/*------------------------------------------------------------------------ Module: ffget.c Author: pldaniels Project: ripmime State: development Creation Date: 14/05/2001 Description: ffget is a small module which will be used to (we hope) speed up the fgetc() routine by line-buffering up first. 12/11/2002: Corrected input buffer termination with fgets where the last line of the input file does not terminate with a \n or \r. 27/09/2001: Added SGI specific compile time changes from char -> short contributed by Don Lafontaine ------------------------------------------------------------------------*/ #include #include #include #include #include #include "logger.h" #include "ffget.h" #define FFGET_VERSION "1.0.0.4" #define FFGET_LASTUPDATED "29/03/2006" #define FFGET_DNORMAL ((FFGET_debug >= _FFGET_DEBUG_NORMAL )) #define FFGET_DPEDANTIC ((FFGET_debug >= _FFGET_DEBUG_PEDANTIC)) /* GLOBALS */ int ffget_linesize=0; int FFGET_doubleCR = 0; int FFGET_SDL_MODE = 0; // Single Char Delimeter int FFGET_SDL_WATCH = 0; // Set if we want to watch for double-CR exploits int FFGET_debug = 0; static char SDL_MODE_DELIMITS[]="\n\r"; static char NORM_MODE_DELIMITS[]="\n"; static char *DELIMITERS=SDL_MODE_DELIMITS; /*------------------------------------------------------------------------ Procedure: FFGET_set_watch_SDL ID:1 Purpose: Set/Unset the flag to indicate that we should be watching out for a double-CR potential exploit mode when decoding files. Input: int level: 0 = don't watch, !0 = watch Output: Errors: ------------------------------------------------------------------------*/ int FFGET_set_watch_SDL( int level ) { FFGET_SDL_WATCH = level; return FFGET_SDL_WATCH; } /*------------------------------------------------------------------------ Procedure: FFGET_set_debug ID:1 Purpose: Set debugging report/verbosity level Input: level Output: Returns the level set Errors: ------------------------------------------------------------------------*/ int FFGET_set_debug( int level ) { FFGET_debug = level; return FFGET_debug; } /*------------------------------------------------------------------------ Procedure: FFGET_getnewblock ID:1 Purpose: Reads a new block of data from the input file Input: FFGET_FILE record Output: Returns number of bytes read Errors: ------------------------------------------------------------------------*/ int FFGET_getnewblock( FFGET_FILE *f ) { int i; int bs = 0; char *p; // We read the maximum of FFGET_BUFFER_MAX -2, because later, when we // use fgets(), we may need to read in an /additional/ single byte // and if we dont allocate spare room, we may have a buffer overflow if (f->FILEEND > 0) { f->endpoint = f->buffer; f->startpoint = f->buffer +1; f->FFEOF = 1; return 0; } else { long block_pos; block_pos = ftell(f->f); bs = fread( f->buffer, 1, _FFGET_BUFFER_MAX -2, f->f ); if (bs < (_FFGET_BUFFER_MAX -2)) { if (feof(f->f)) { f->FILEEND = 1; } else { fprintf(stderr,"%s:%d: WARNING - File read failed with error:%s",__FILE__,__LINE__,strerror(errno)); return 0; } } if (bs > 0) { // If we read in some data, then adjust the buffer to deal with it // // First we set the start point back to the start of the buffer, // then we set the end point to be the start +datasize we read, -1 // then we adjust the total bytes read (for the sake of record keeping // though it has no /real/ purpose) // f->last_block_read_from = block_pos; f->startpoint = f->buffer; f->endpoint = f->startpoint +bs -1; f->bytes += bs; // Check the buffer for poisioning \0's // As these routines are being used for 7-bit valid text data, // we have to filter out any nasty \0's. p = f->startpoint; for (i = 0; i < bs; i++) { if (*p == '\0') *p = ' '; p++; } *p = '\0'; if (FFGET_DPEDANTIC) LOGGER_log("Size: %ld bytes\n",f->bytes); } } return bs; } /*------------------------------------------------------------------------ Procedure: FFGET_presetbuffer ID:1 Purpose: Presets the FFGET buffer with defined data Input: FFGET_FILE record Buffer to get data from Quantity of data Output: None Errors: ------------------------------------------------------------------------*/ int FFGET_presetbuffer( FFGET_FILE *f, char *buffer, int size ) { if (size > _FFGET_BUFFER_MAX) size = _FFGET_BUFFER_MAX; memcpy(f->buffer,buffer,size); f->startpoint = buffer; f->endpoint = buffer +size; return 0; } /*------------------------------------------------------------------------ Procedure: FFGET_setstream ID:1 Purpose: Sets the FILE * stream to the FFGET_FILE record Input: FFGET_FILE record Stream to use. Output: Errors: ------------------------------------------------------------------------*/ int FFGET_setstream( FFGET_FILE *f, FILE *fi ) { f->f = fi; f->bytes = 0; f->linecount = 0; f->endpoint = f->buffer; f->startpoint = f->endpoint +1; f->trueblank = 0; f->ungetcset = 0; f->lastchar = '\0'; memset(f->buffer,'\0',_FFGET_BUFFER_MAX +1); f->c = '\0'; f->FFEOF = 0; f->FILEEND = 0; f->last_block_read_from = -1; return 0; } /*------------------------------------------------------------------------ Procedure: FFGET_closestream ID:1 Purpose: Closes the stream contained in a FFGET record Input: FFGET record containing the stream to close. Output: Errors: ------------------------------------------------------------------------*/ int FFGET_closestream( FFGET_FILE *f ) { f->startpoint = f->endpoint = NULL; return 0; } /*------------------------------------------------------------------------ Procedure: FFGET_feof ID:1 Purpose: Returns the status of FFGET's EOF Input: FFGET record Output: EOF status (0 == NOT eof, 1 == EOF has been reached) Errors: ------------------------------------------------------------------------*/ int FFGET_feof( FFGET_FILE *f ) { return f->FFEOF; } /*-----------------------------------------------------------------\ Function Name : FFGET_seek Returns Type : int ----Parameter List 1. FFGET_FILE *f, 2. size_t offset , ------------------ Exit Codes : Side Effects : -------------------------------------------------------------------- Comments: Seeks to 'offset' bytes from the first byte of the file. Reloads the buffer block. -------------------------------------------------------------------- Changes: \------------------------------------------------------------------*/ int FFGET_seek( FFGET_FILE *f, long offset, int whence ) { int result = 0; fseek(f->f, offset, whence); result = FFGET_getnewblock(f); return result; } /*-----------------------------------------------------------------\ Function Name : FFGET_tell Returns Type : size_t ----Parameter List 1. FFGET_FILE *f , ------------------ Exit Codes : Side Effects : -------------------------------------------------------------------- Comments: Returns the position in the file that the current "file cursor" is pointing to. -------------------------------------------------------------------- Changes: \------------------------------------------------------------------*/ long FFGET_ftell( FFGET_FILE *f ) { long pos; pos = f->last_block_read_from +(f->startpoint -f->buffer); return pos; } /*------------------------------------------------------------------------ Procedure: FFGET_ungetc ID:1 Purpose: Pushes back into the buffer (effectively) a single character Input: FFGET record Character to retain for the next read. Output: Errors: ------------------------------------------------------------------------*/ int FFGET_ungetc( FFGET_FILE *f, char c ) { f->c = c; f->ungetcset = 1; return 0; } /*------------------------------------------------------------------------ Procedure: FFGET_getc ID:1 Purpose: Gets a single character from the FFGET buffer Input: FFGET record Output: Single character from the buffer, or EOF if end of file. Errors: ------------------------------------------------------------------------*/ #ifdef sgi short FFGET_fgetc( FFGET_FILE *f ) #else char FFGET_fgetc( FFGET_FILE *f ) #endif { int c; if (f->ungetcset) { f->ungetcset = 0; return f->c; } if ((!f->startpoint)||(f->startpoint > f->endpoint)) { FFGET_getnewblock(f); } if (f->FFEOF == 0) { c = *f->startpoint; f->startpoint++; } else { c = EOF; } return c; } /*------------------------------------------------------------------------ Procedure: FFGET_fgets ID:1 Purpose: Gets a single line from the input buffer. The line can be either \r \n \r\n terminated based on the status flags set/unset by previous reads. This function is the key to making tools like ripMIME be able to see double-vision, that is, to see emails like Outlook does, and also like RFC. Input: line: Buffer to write to max_size: Maximum number of bytes to write to line. f: FFGET record to use to read. Output: Pointer to line. Errors: ------------------------------------------------------------------------*/ char *FFGET_fgets( char *linein, int maxsize, FFGET_FILE *f ) { char *line = linein; char *crlfpos; int charstoCRLF; int chardiff; int result; int max_size = maxsize; // We /ONLY/ return EOF -here-, because, even if during the process of // reading through the bufffer we reach the end, we do still have data // which we can send back to the caller. I'm surprised that more people // didn't pick up on this, it's a glaring flaw. f->trueblank = 0; if (f->FFEOF != 0) { return NULL; } if ((FFGET_SDL_WATCH > 0)||(FFGET_SDL_MODE != 0)) { DELIMITERS = SDL_MODE_DELIMITS; } else DELIMITERS = NORM_MODE_DELIMITS; // fprintf(stderr,"FFGET_called, SDLMODE = %d, Offset = %d, maxsize = %d, DATA left = %d, first char is '%02X'\n", FFGET_SDL_MODE, (f->startpoint -f->buffer), max_size, (f->endpoint -f->startpoint)+1, (int)(*f->startpoint)); max_size = maxsize = maxsize -2; // memset(line, 0, max_size+1); // If we dont have enough data in the buffer to fill up the fgets request // we'll have to do a two step fill //fprintf(stderr,"DATA Reminianing : %d\n", (int)(f->endpoint -f->startpoint)+1); if ((f->startpoint > f->endpoint)) { result = FFGET_getnewblock(f); if (result == 0) { *linein = '\0'; return NULL; } } // This loop does not go around too many times, once, maybe twice max. while ((max_size > 0)&&(f->FFEOF == 0)) { crlfpos = strpbrk( f->startpoint, DELIMITERS); if (crlfpos) { // if our next char is a CR or LF, then pick it up and // return it with the line. NOTE - this is to deal with // CRLF pairings which are common on DOS files. In fact, // this is a case of where UNIX is actually -wrong-. It // should have also used CRLF pairing to mark line ends, but // someone obviously (and understandably, to save space) // thought they'd leave make LF imply a CR as well. // Well done... another bugger up in life. // The logic of this nested IF statement is as follows... // If we do have another char available... // and if the pairing is not \n\n (which should be treated as two lines // and if the next char is a \n or a \r, // THEN we should increment the end of line pointer so that we // include the additional \n or \r into the line we're going to // return // If we are NOT in the Single-delimeter mode (SDL_MODE), and the next // char is available, then commence the delimeter testing routines if ((0==f->FILEEND)&&(0==f->FFEOF)&&( ((crlfpos +1) > f->endpoint))) { // We have an EOL character, get 1 more from the stream to test the next character int c; c = fgetc(f->f); if (c==EOF) { // fprintf(stderr,"EOF hit due to fgetc()\n"); f->FILEEND = 1; } else { if (c == '\0') c = ' '; // Check for character value vadality if ((c > 0) && (c <= 255)) { f->endpoint++; *(f->endpoint) = c; *(f->endpoint+1) = '\0'; } } } // If (crlfpos +1) is /not/ within our buffer bounds // If the next char from our CRLF pickup is within the bounds of // our endpoint, then proceed to test the CRLF combo if ( ((crlfpos +1) <= f->endpoint)) { // fprintf(stderr,"Found '%02X' [next is '%02X']\n",*crlfpos, *(crlfpos+1)); if ( *crlfpos == '\n' ) { if ( *(crlfpos +1) == '\r' ) { crlfpos++; } } // If our combo starts with a \r, then test it to see // if we have another \r after it, in which case, we // turn on SINGLE_DELIMETER_MODE. if ( (*crlfpos == '\r') ) { if ( *(crlfpos +1) == '\r' ) { // A \r\r sequence has just been detected, set our doubleCR flag // so that MIME_headers can read it and react accordingly. // Look out for single \r's from here on, as they are now seen as // EOL markers in Outlook. FFGET_doubleCR=1; FFGET_SDL_MODE=1; crlfpos++; } else if ( *(crlfpos +1) == '\n' ) { // If we see a \n after our \r, then treat this as a single // line delimeter if we are NOT in Single Delimeter mode if (!FFGET_SDL_MODE) { crlfpos++; } } else { // If we saw a \r, but then there was no other EOL type char (\r or \n) // then switch to SDL mode (Single delimeter). FFGET_SDL_MODE=1; } } // If combo starts with a \r } // If crlfpos +1 is within the bounds of our buffer. // Determine how many characters/bytes there are from the startpoint, // to the CRLF position. charstoCRLF = crlfpos -f->startpoint; // If the number of chars is -less- than that of the maximum line read // size which our calling program has specified, then we set the max_size // to be the number of chars. //DEBUG fprintf(stderr, "MAX_size = %d, charstoCRLF = %d\n", max_size, charstoCRLF); if ((charstoCRLF >= 0)&&(charstoCRLF < max_size)) max_size = charstoCRLF; } // If CRLF pos found. // else crlfpos = (f->endpoint +1); // If the buffer amount remaining in our FFGET buffer is greater than // the maximum size available in our line buffer, then we // only copy the max_size amount across if (( f->endpoint -f->startpoint) >= max_size) { if (max_size < 0) fprintf(stderr,"ERROR - Max size < 0\n"); memcpy(line, f->startpoint, max_size +1);//+1 f->startpoint += (max_size +1); //+1 *(line +max_size +1) = '\0'; //+1 max_size = 0; } else { // else, if the amount of data available is /LESS/ than what we can // accept in the line buffer then copy what remains out to the line // buffer and then tack on the results of a new read. chardiff = f->endpoint -f->startpoint; // fprintf(stderr,"CHARDiff = %d, FFEOF = %d, FILEEND = %d\n",chardiff, f->FFEOF, f->FILEEND); if (chardiff >= 0) { memcpy(line, f->startpoint, chardiff +1); *(line +chardiff +1) = '\0'; // 12-11-2002: Added this line to terminate the input buffer incase it wasn't already flushed with \0's line += (chardiff +1); max_size -= (chardiff +1); f->startpoint = f->endpoint +1; if (max_size < 0) max_size = 0; } FFGET_getnewblock(f); } // If there wasn't enough data to satisfy ends. } // While we've got space to fill, and we've got data to read line = linein; f->trueblank = 0; if ((f->lastchar == '\n')||(f->lastchar == '\r')) { if ((line[0] == '\n')||(line[0] == '\r')) { f->trueblank = 1; } } f->lastchar = line[strlen(line) -1]; f->linecount++; return linein; } /*------------------------------------------------------------------------ Procedure: FFGET_raw ID:1 Purpose: This is a hybrid binary-read and fgets type read. This function reads data from the input buffer until it encounters a \r \n \r\n at which point it will return to the calling parent with its buffer containing that line. This is required so that we dont miss any boundary specifiers which are located on new-lines. Input: f: FFGET record buffer: memory location to write data to max: maximum holding capacity of the raw buffer Output: Returns the number of bytes placed into the buffer. Errors: ------------------------------------------------------------------------*/ int FFGET_raw( FFGET_FILE *f, unsigned char *buffer, int max ) { unsigned char c; // read buffer int bytestogo = 0; int count = 0; // How many bytes read // Special situation here, if we have a return from MIME_headers which indicates // that we have data in a MIMEH_pushback, then we need to process that first, before we // go back into the data file. if ((!f->startpoint)||(f->startpoint > f->endpoint)) { bytestogo = FFGET_getnewblock(f); } else { bytestogo = f->endpoint -f->startpoint +1; } // Whilst we've got less bytes than the maximum availabe // for the buffer, we keep on reading // while (count < max) { if (!bytestogo) { bytestogo = FFGET_getnewblock(f); } if (!f->FFEOF) { c = *f->startpoint; f->startpoint++; *buffer = c; buffer++; count++; bytestogo--; // If we get a line delimeter, check to see that the next char (which is now // pointed to at f->startpoint isn't a delimeter as well which perhaps we should // be including in our line were' going to return // // 25/05/02 - Silly mistake, I had (!\n || !\r) when it should be && (ie, if the next // char is NEITHER of the \n or \r chars, then break. // if ((c == '\n')||(c == '\r')) { if ( (*(f->startpoint) != '\n') && (*(f->startpoint) != '\r') ) break; } } else break; } *buffer = '\0'; return count; } //--------------END.