Advanced C Programming

Spring 2024 ECE 26400 :: Purdue University

This is a PAST SEMESTER (Spring 2024).
Due 1/16

Editing: Vim

Goals

This assignment has the following goals.

  1. Learn to use the Vim text editor to efficiently move around a code file and make changes.
  2. Practice improving the style of poorly formatted code.
  3. Get to know how grades will be calculated in this course.

Setup

In this course, you will work on the ecegrid server. That's where you will get your starter files (if any), pre-test your submission (when possible), and submit your code. This gives everyone the same environment, with the same version of the tools (e.g., gcc, vim, etc.), and the same platform for testing code.

Before you proceed, please complete the setup.

Start learning Vim

This assignment is your chance to get good at editing code in Vim, before we get to the more challenging C programming assignments. You should take this opportunity to learn it well.

Bare minimum: Go through vimtutor

vimtutor is an interactive tutorial that is installed on ecegrid and every computer where Vim is installed. At the command line, just enter vimtutor and follow the instructions.

This will take about 15-30 minutes, and will make the rest of this much easier. The Resources page has some web-based options, which look nicer but are less complete.

Recommended: Print out a reference sheet (or two or three)

Recommended: Watch a video (or two or three)

Get the homework files

To fetch the files for this assignment, enter 264get hw01 from bash. That should create a new directory called hw01. To see the directory enter ls . Next, enter cd hw01 to enter the directory and start working.

Do the homework

Here's the scenario: You have been given a partially-working calculator for ECE 264 grades. However, the code is very messy. First, you will fix the formatting. Next, you will complete the code. Finally, you will finish the code so that it prints out a course grade.

For this assignment, we will record the keys you type into vim. When the assignment is scored, we will play the keys back to verify if you reached the intended result. You will start with before.c and finish with goal.c. Do not modify goal.c or before.c. You may need those.

Do not use the arrow keys or mouse. As with any tool, your coding will be most efficient if you use Vim in the way that it was designed (e.g., j for down, k for up, etc.). For this assignment, you are required to work in this way.

We will give you hints in purple for some steps. For the others, you will need to use what you learned from vimtutor and/or the reference sheet. In this explanation, <esc> refers to the Escape key, <enter> refers to the Enter key, <tab> refers to the Tab key, and <space> refers to the spacebar.

Tips

One session. You must do the entire process in one vim session.

Repetition. If you exit vim before the files match, you will need to start over. For that reason, you may end up doing the whole process several times. The repitition will serve as good practice. Once you are very comfortable with vim and understand what you are doing, it shouldn't take more than about 10 minutes to edit before.c into goal.c.

⚠ Pitfalls

Do not use the arrow keys or mouse. You will not receive credit if you do. To avoid accidentally using them, enter the following command in vim (from normal mode): :DisableMouseAndArrowKeys and press enter. Do not depend on that command completely as it might not catch all arrow keys.

Spacing. Pay close attention to the spacing in goal.c. For example, all of the // hw# and // exam# comments start in column 11. The comma in those lines is in column 8. Vim shows the current line and column in the lower-right corner of your screen.

Splits. Opening a split while editing your after.c and recording your recorded.vim will probably work out okay, but it isn't recommended. We suspect it might lead to inconsistencies when the keystrokes are played back, but we haven't verified this. If your recorded.vim is passing the HW01 pretester (264test_special_vim), then you should get 100% credit.

-W not -w. When typing vim -W recorded.vim -n -i NONE after.c , the -W must be uppercase (i.e., -W not -w).

-W only for recording. If you want to look at the files after you are done, be sure to not use the -W recorded.vim argument to vim. Otherwise, you will overwrite your recorded.vim and you will have to start over.

File size. Your recorded.vim will be likely be >1500 bytes, and probably much more. If it is very small (e.g., 0 bytes), you probably overwrote it. (Type ls -l from bash to see the file sizes.)

Get the homework files

From bash, enter cp before.c after.c to make a copy of the pristine starter file.

Next, enter vim -W recorded.vim -n -i NONE after.c to open it in vim.

The extra parameters (-W recorded.vim -n -i NONE) are only for this assignment.

For those who are curious… The -W recorded.vim parameter makes it record your keystrokes. The -i NONE parameter tells Vim to start fresh (i.e., forget macros and marks leftover from previous sessions). That way, when we replay your session, we will get the same result you got. The -n parameter turns off the swap file.

  1. Fix the missing semicolon on line 4.
    1. Move to line 4. 4G
    2. Append a semicolon at the end. A;<esc>
  2. Indent the body (everything inside the curly braces) of the average function, which starts on line 1215.
    1. Search for "average".    /average<enter>
    2. Go down one line (to get to the beginning of the body of this function). j
    3. Select the current line using visual mode. V
    4. Extend the selection to include the rest of the body of this function. jjjj
    5. Indent everything that is selected. >
  3. Indent the body of the for loop inside the average function.
    1. Move to line 19 16.
    2. Indent the current line. >>
  4. Indent the bodies of the weighted_sum, weighted_average, and main functions, as well as any for loops inside them.
  5. Add a blank line after the second #include statement at the top.
    1. Jump to the beginning of the file. gg
    2. Move your cursor to beginning of the second #include statement. jj
    3. Start a blank line below the current line. o (lowercase letter)
    4. Exit insert mode (to get back to normal mode) <esc>
  6. Add a blank line between the declarations of HOMEWORK_WEIGHTS and NUM_EXAMS.
  7. Add a blank line above the declaration of the average function.
    1. Search for "average".
    2. Enter a blank line above the current line. O (uppercase letter)
    3. Exit insert mode (to get back to normal mode).
  8. Ensure that there is exactly one blank line above all function declarations.
  9. In the const arrays declared at the top of the file (HOMEWORK_SCORES, HOMEWORK_WEIGHTS, EXAM_SCORES, EXAM_WEIGHTS), put each number on its own line.
    1. Jump to the top of the file.
    2. Move down to line 6 5.
    3. Move to the beginning of the line. 0 (zero)
    4. Move right by 4 words. wwww
    5. Insert a carriage return, and then immediately exit insert mode (back to normal mode) i<enter><esc>
    6. Start creating a quick macro so you don't have to do that so many times. qq
    7. Move right by words until the cursor is at the beginning of a number. www
    8. Insert a carriage return and then immediately exit insert mode (back to normal mode).
    9. Stop recording the quick macro. q
    10. Replay the macro once. @q
    11. Replay the same macro several more times. @@@@@@@@@@@@ (…and so on)
    12. Insert a carriage return just before the ending curly brace (}) and ensure that it is flush with the left side (i.e., not indented).
    13. Follow a similar procedure for HOMEWORK_WEIGHTS, EXAM_SCORES, and EXAM_WEIGHTS.
  10. Right align all of the numbers.
    1. Move to the first 2-digit number in HOMEWORK_SCORES (70).
    2. Insert one space before the line beginning. I<space>
    3. Go down one line (to get to the number 80).
    4. Repeat the last action. (period)
    5. Do likewise for the rest of the two-digit numbers in HOMEWORK_SCORES.
  11. Add a comment after every number in each of the const arrays declared at the top of the file (HOMEWORK_SCORES, HOMEWORK_WEIGHTS, EXAM_SCORES, EXAM_WEIGHTS), to indicate which homework or exam they refer to.
    1. Move to the first number in HOMEWORK_SCORES.
    2. Append the text " // hw" at the end of the line. A<space><space>//<space>hw<esc>
    3. Move back to the first number in HOMEWORK_SCORES.
    4. Append the number "1" at the end of the line so that the comment reads " // hw1".
    5. Append the numbers "2" to "15" at the end of the remaining lines in HOMEWORK_SCORES.
    6. Follow a similar procedure for HOMEWORK_WEIGHTS, EXAM_SCORES, and EXAM_WEIGHTS.
  12. Ensure that there is exactly one space after every comma in a function parameter list.
  13. Ensure that there is exactly one space before and after every arithmetic operator, as shown in goal.c.
  14. For the very long arithmetic expressions (exam_score, and summary_score), break them into two lines but align the equals sign in the first line with the plus sign in the second line, as shown in goal.c.
  15. There is a bug. The print_grade(…) function is used in main(…) but is never defined. Add the following function just above main(…):
    void print_grade(float summary_score) {
      printf("summary score:  %.2f\n", summary_score);
    }
  16. Save. :w<enter>
  17. Modify the print_grade(…) function so that instead of printing the summary score, it prints the letter grade. You will need an if statement (or if else) for each possible letter grade. Do not type all of these repeatedly. Instead, you will use copy-paste. (In vim parlance, these are called yank and put.)
    1. Move to the definition of the print_grade(…) function.
    2. Move to the end of the line. $
    3. Move to the corresponding closing curly brace. %
    4. Move up one line.
    5. Delete the current line ("printf(…"). You should now be back on the closing curly brace.
    6. Add a line above this closing curly brace. This is where you will enter the code to print the letter grade.
    7. Type the following code.
      if(summary_score >= 85) {
          printf("A\n");
      }
    8. Select the ending curly brace using visual mode.
    9. Extend the selection upward to include all 3 lines of this if statement and its body.
    10. Yank (copy) the selection. y
    11. Move back to the ending curly brace of the if statement.
    12. Put (paste) the text that you previously yanked directly below the if statement. p
    13. Insert the text "else " at the beginning of this line.
    14. Change "85" to "82" and change "A" to "A-" so that you have the following:
      if(summary_score >= 85) {
          printf("A\n");
      }
      else if(summary_score >= 82) {
          printf("A-\n");
      }
    15. Add three copies of the else if clause and adjust the score and grade letter so that you have the following:
      if(summary_score >= 85) {
          printf("A\n");
      }
      else if(summary_score >= 82) {
          printf("A-\n");
      }
      else if(summary_score >= 78) {
          printf("B+\n");
      }
      else if(summary_score >= 75) {
          printf("B\n");
      }
      else if(summary_score >= 72) {
          printf("B-\n");
      }
    16. Using a similar method, copy these 9 lines to insert the code for "C+", "C", and "C-". Repeat for "D+", "D", "D-".
    17. Add the else clause for the “F” grade.
  18. Fix the two for loops so that they conform to the Code Quality Standards, especially with respect to curly braces and for loop counter scope
  19. Make any other changes needed so that your after.c file is identical to goal.c.
  20. Save.
  21. Exit vim. :q<enter>

Test

There is a special pretester for HW01. From your hw01 directory, type 264test_special_vim . That will replay your keystrokes (recorded.vim) in vim and display any differences with goal.c. If there are no differences, then you are done. Your task is to edit before.c after.c to make it identical to goal.c. If it prints a "Congratulations..." message, you should receive 100%, once scores are posted (probably next week).

HW01 is special in that it is the only homework that offers a pretester with a guarantee. On all future assignments, you will be solely responsible for testing your code. If a pretester is offered, it may be only a partial check, and it may not be available all the time.

Submit (or save)

In general, to submit any assignment for this course, you will use the following command:

264submit ASSIGNMENT FILES…

For HW01, you will type 264submit hw01 recorded.vim from inside your hw01 directory.

You can submit as often as you want, even if you are not finished with the assignment. That saves a backup copy which we can retrieve for you if you ever have a problem.

Before you consider this done, you may want to look over the section on pitfalls above.

How much work is this?

For those who do not know vim (or don't know it very well), we estimate it may take about 2 hours to go through the tutorial(s) and another 2 hours to edit the before.c to make it identical to goal.c, including time for several repetitions.

If you already know vim well, this assignment should be very easy.

Q&A

Doing this assignment

  1. How do I know if I have used any arrow keys?

    The pretester (264test_special_vim) will tell you.

  2. I made a tiny mistake. How can I fix it?

    Is it best to start over. The practice will help you get more comfortable editing code with Vim.

  3. I accidentally pressed arrow keys. What should I do?

    Start over.

  4. How do I start over?

    Make a fresh copy of before.c like this:

    you@eceprog ~/264/hw01 $ cp before.c after.c

    Then open Vim using the special flags for HW01:

    you@eceprog ~/264/hw01 $ vim -W recorded.vim -n -i NONE after.c
  5. Isn't there an easier way than starting over?

    There are some hacky ways to recover if you made a small mistake, but they are not supported because they won't save much time (if any) and they won't teach you anything useful.

    Repeating the process in this assignment will give you additional practice and help you develop muscle memory for the commands.

  6. What if I accidentally click the mouse in the window?

    Press gg right afterward to put the cursor on the top line. Then continue. That will make sure that accidentally repositioning the cursor with the mouse doesn't throw off the line that you are editing.
  7. If I quit and come back, can I start where I left off?

    Short answer: No.

    You should do the whole process in one session.

    If you quit, it will leave a :q (or similar) command in your recorded.vim file. We looked into possibly filtering those out, but it gets complicated to do properly. Editing your recorded.vim file to remove the :q<enter> is a possibility. You might try doing :vert diffsplit goal.c to see differences and fix them. (You would open your after.c and then enter that command from inside Vim.)

  8. How can I see what my recorded.vim contains?

    Run this utility.

    you@eceprog ~/264/hw01 $ vim_pretty_print recorded.vim

    Note: vim_pretty_print is not a standard utility. Prof. Quinn made it to help with this assignment.

  9. May I edit my recorded.vim file directly?

    It is not recommended and not supported.

    Editing code proficiently in Vim is a skill that will be an asset to you as a programmer in the future. Fiddling with the recorded.vim will be useful to nobody, beyond this assignment.

    The script you turn in must be able to transform before.c into goal.c. If you can find a way to edit your script to make that work, that's fine. However, it may be easiest to just start over. Once you've learned the commands that this assignment uses, the process of transforming before.c into goal.c should be relatively quick.

Scoring

  1. How will this be scored?

    We will run it with our tester, which works in a very similar way to the pretester (264test_special_vim). It starts with the pristine before.c—just as you received when you ran 264get. It then copies before.c to after.c, just as you did when you ran the cp before.c after.c command in bash. Next, it opens Vim and replays your keystrokes using the recorded.vim that you submitted. That should transform before.c to be identical to goal.c. Finally, the tester compares the resulting after.c to a pristine goal.c, line by line. If they are identical, you get 100% credit.
    We might forgive very minor differences, but do not count on that, and do not ask course staff what will be forgiven. Your job is to make them identical.
  2. How can I be sure I will get 100%?

    If the HW01 pretester (264test_special_vim) prints “Congratulations” (in green), you should receive 100% credit. This is special for HW01.
    Disclaimer: This assumes you submit your recorded.vim—the same one that was in your directory when you ran the pretester—with no further modifications, and that you did not modify before.c or .vimrc files.
    HW01 is special in that it is the only homework that offers a pretester with a guarantee. On all future assignments, you will be solely responsible for testing your code. If a pretester is offered, it may be only a partial check, and it may not be available all the time.
  3. What if I get the “Congratulations” (in green) but I do not receive 100% credit?

    It is unlikely that our tester would give a different result from the pretester you were given (264test_special_vim), but if it happens, we will honor the result from the pretester. (Note: This is the only pretester that offers such a guarantee.)

    You will have an opportunity to submit a regrade request. We will then run the pretester you were given (264test_special_vim) in a directory with a pristine before.c and the same recorded.vim that you submitted. Alternative, we may ask you to come to office hours and do the same from your account. If it prints “Congratulations” (in green), then we will make sure you get 100% credit.
    If it does not print “Congratulations” (in green), we will not negotiate the amount of partial credit (if any). Please submit a regrade request only if you got your after.c identical to goal.c, as evidenced by the “Congratulations” (in green) message.
  4. What if I get it close, but it does not print “Congratulations” (in green)?

    We may assign partial credit, based on the number of lines that matched. A line that matches partially—i.e., different by a few characters—is not a match.
    We will not negotiate the right amount of partial credit. Do not submit a regrade request asking for more partial credit.

About Vim

  1. What is vim?

    Vim is a code editor (a text editor designed for programming).
  2. What's the difference between vim and vi?

    Typing vi gives you vim in a "compatibility mode" that simply turns off many of the newer features. It is actually the same program. There is no reason to type vi. Type vim instead.
  3. Where can I use it?

    Vim is almost always installed on Linux and UNIX machines. On Windows, you can install gVim, which gives you GUI.
  4. Can I use the mouse with Vim?

    Yes. This works in PuTTY, but may not work in other SSH programs. (This sample .vimrc enables this for you. If you are using your own .vimrc file, you need to add set mouse+=a to it.)
  5. Can I use tabs?

    Yes. To open several files using tabs, use the -p flag at the command line (e.g., vim -p file1.c file2.c …). Within Vim, use the :tabe command to open another file in a separate tabe (e.g., :tabe file2.c). With the sample .vimrc file, you can also just press F2 to open a new tab, [ to move to a tab to the left, and ] to move to the tab to the right.
  6. Can I get syntax highlighting?

    Yes. (The sample .vimrc file turns this on for you.)

Purpose of this assignment

  1. I already know an editor. Why should I learn Vim?

    Even if you know and love some other editor, learning Vim will make you a more versatile programmer. Unlike most other editors, Vim can be used through a terminal (e.g., PuTTY, ssh), which will make your life a lot easier for all homework assignments in this course. this course

    In ECE 264, you will find the workflow for homework assignments a lot simpler and less error-prone if you work directly on ecegrid through an SSH terminal. The course is optimized for working in this way. You will fetch starter files, test your code, and submit—all through an SSH window. Although it would be possible to write your code in another editor on your computer (e.g., Sublime, TextMate, etc.), and transfer the files to ecegrid, that would be inefficient and error-prone. Once you are comfortable editing code through an SSH terminal, you will not need to transfer any files anywhere. Out in the world, that comes in handy when editing code that runs on a server, as well as some embedded systems.

  2. Why not Emacs?

    Emacs is the only other viable choice for coding in a terminal. Like Vim, it is a very powerful tool for programmers who take the time to learn it well.

    We chose Vim for this course because (a) it is introduced lightly in CS 159, (b) its key replay features makes this assignment possible, and (c) it is available and usually pre-installed on practically every Linux/UNIX-based system in the world.

    For decades, there was an ongoing debate over which of the two was better. (Search for [editor war] or [vim vs. emacs], or just read this great article on Slate.) In recent years, several online polls (here, here, and here) have found Vim to be more “popular”, by a ratio of around 4 to 1. However, both have unique features and virtues that appeal to different people. Those who prefer Emacs have plenty of great reasons to do so. The choice of Vim for this class was borne of pragmatism.

  3. Why are we learning any editor? Why not just focus on C?

    This class is about “Advanced C Programming (verb), not “The C Language” (noun). As such, the focus is on doing. When you finish this course, you will be expected to know how to write programs. All serious code editors require some learning. In the past, we found that many ECE students were so focused on the assignments that they never took the time to learn any tool well. We created this assignment to ensure that everyone knows at least one editor well enough to be productive. Typically, very few students come in know some other editor well. Those that do will still benefit, by becoming ambidextrous, and will most likely find this assignment easy anyway.
  4. Do I always have to use vim?

    No. Everyone must learn Vim, and use it for this assignment. After that, you are free to use whatever you like.
  5. Are there any drawbacks to using some other editor?

    Yes. If you use a local editor (i.e., running on your laptop or desktop computer), you will need to transfer the starter files from ecegrid to your computer. Then, every time you want to test them, you will need to transfer your code back to ecegrid. Your code must compile and run on ecegrid using our version of the gcc, so it is important to test on ecegrid. Finally, when you are ready to submit, you will again need to transfer your files.

    If you already know Emacs well, there is no drawback to using that for future assignments. Emacs is installed on ecegrid, and works well through an SSH terminal.

  6. How can I possibly learn to use Vim while simultaneously solving challenging programming assignments?

    This assignment is for you. Learning to use Vim—or any other serious programming editor—well takes effort, so we are giving you this assignment to focus on just that. After this, you will move on to C programming. (There will be one more tool-focused assignment, in which you will learn the Gdb debugger.)

Trivia

  1. What do the codes mean that I see in recorded.vim?

    In helping some students troubleshoot HW01, we have learned a bit about the internal workings of Vim and these strange key logs (i.e., recorded.vim. We are sharing this, in case anybody is curious. This is trivia. You do not need to read or understand anything in this Q&A entry.
    That said, here are what most of the codes mean.
    <80>ku ↑ up arrow
    <80>kd ↓ down arrow
    <80>kl ← left arrow
    <80>kr → right arrow
    ^M Enter key
    ^] Esc key
    ^I Tab key (use set :list to show these
    <80>kI Ins key
    <80>kN PgDn key
    <80>kP PgUp key
    <80>kb Backspace key
    <80>k1 F1 key
    <80>k2 F2 key
    <80>k3 F3 key
    <80>k4 F4 key
    <80>k5 F5 key
    <80>k6 F6 key
    <80>k7 F7 key
    <80>k8 F8 key
    <80>k9 F9 key
    <80>k2 F2 key
    <80>k3 F3 key
    <80>k4 F4 key
    <80>k5 F5 key
    <80>k6 F6 key
    <80>k7 F7 key
    <80>k8 F8 key
    <80>k9 F9 key
    <80>k; F10 key
    <80>F1 F11 key
    <80>F2 F12 key
    <80><fd> ignore − probably internal window events
    <80>ý▒ ignore − probably internal window events
    When reading one of these, it is best to run two commands:
    :s/Ctrl v80Ctrl vfd.//g
    :set list
    The first one strips out the irrelevant window event codes. They do not correspond to keys or mouse actions. They just make the file hard to read. The second command tells Vim to show Tab keys as ^I. Otherwise, they may show as just a few spaces.
    If you are trying to understand why a recorded.vim isn't working as you expected, it helps to strip out the window event codes (e.g., <80><fd>`). Make a “clean” copy of your recorded.vim without them using the sed command.
    From bash:
    sed '/\x80\xfd.//g' < recorded.vim > clean.vim
    
    Next, submit and make a backup copy of your original recorded.vim.
    264submit hw01 recorded.vim
    cp recorded.vim orig_fn
    Now, you can make a shorter version of recorded.vim using the head command, to see where the problem started. This command copies all but the last 50 characters of clean.vim to recorded.vim.
    head -c -50 > recorded.vim

Updates

1/14/2020 Fixed line numbers (6 → 5; 19 → 16).
1/16/2020 Added warning that -W is capital, not lowercase.
Added a section about Pitfalls.
Added Q&A items about scoring.