C fgetc sigsegv - skipping to end of line

328 Views Asked by At

I'm reading a file char by char. When I reach a colon I want to skip past all characters until I reach a newline character. Effectively, after seeing a colon I wish to skip to the next line (if another line exists). Seems simple enough, but I'm receiving a sigsegv when I try to break out of my second while loop which skips over the data I don't care about.

Without the break the code behaves as I would expect it to (although not my desired output). That is, it reads data until the first colon, then it will skip to EOF and exit.

      5    FILE *fp = fopen("myFile", "r");
      6    char *string = (char*) malloc(sizeof(char));
      7    char *head = string;
      8  if( fp ){
      9       int c;
     10       while( (c=fgetc(fp)) != EOF ){
     11          if(c == ':'){
     12             *string++ = '\n';
     13             while( (c=fgetc(fp)) != EOF ){
     14                if( c == '\n' )             // Skip to end of line
     15                   break; //removing break avoids sigsegv
     16             }
     17          }else{
     18             *string++ = c;
     19          }
     20       }
     21    }

It seems that when I break out of the loop, either c or fp are modified somehow that cause a sigsegv. My best guess is that fp is somehow modified and generates this error when the parent fgetc() calls on fp. Beyond that, though, I'm not sure what's causing the issue.

2

There are 2 best solutions below

1
On

You need to allocate more bytes for string. This line:

char *string = (char*) malloc(sizeof(char));

is only allocating one byte for your string.

0
On
the main problem was writing to offset from a pointer
that did not point anywhere in particular
which is a major reason for getting seg fault events
note: 'string' is a well known C++ class name
      so changed the name to pBuffer
The following code snippet fixes that problem (and a few others)


    int   memorySize = 1024;
    char *realloc_result = NULL;
    char *pBuffer = malloc(memorySize);
    if( NULL == pBuffer )
    { // then, malloc failed
        perror( "malloc failed" );
        exit( EXIT_FAILURE );
    }

    // implied else, malloc successful

    FILE *fp = fopen("myFile", "r");
    if( NULL == fp )
    { // then fopen failed
        perror( "fopen failed for read" );
        fclose(fp);
        exit( EXIT_FAILURE );
    }

    // implied else, fopen successful


    int c;     // input buffer char
    int i = 0; // loop counter
    while( EOF != (c=fgetc(fp)) )
    {
        if( i >= memorySize )
        { // then need more memory

            if( NULL == (realloc_result = realloc( pBuffer, memorySize*2 ) )
            { // then realloc failed

                perror( "realloc failed" );
                free(pBuffer);
                fclose(fp);
                exit( EXIT_FAILURE );
            }

            // implied else, realloc successful

            memorySize *= 2; // track current amount of allocated memory
            pBuffer = realloc_result; // update ptr to allocated memory
        } // end if

        if( ':' == c )
        { // then, terminate buffered line and skip to end of input line

            pBuffer[i] = '\n';
            i++;

            // skip to end of line
            while( EOF != (c=fgetc(fp)) )
            {
                if( '\n' == c )
                { // then, reached end of input line
                    break; // exit inner while
                } // endif
            } // end while

            if( EOF == c )
            { // then, at end of file
                break; // exit outer while
            } // end if
        }

        else
        { // not skipping data
            pBuffer[i] = c;
            i++;
        } // end if
    } // end while