/*
** Splint - annotation-assisted static program checker
** Copyright (C) 1994-2003 University of Virginia,
** Massachusetts Institute of Technology
**
** This program is free software; you can redistribute it and/or modify it
** under the terms of the GNU General Public License as published by the
** Free Software Foundation; either version 2 of the License, or (at your
** option) any later version.
**
** This program is distributed in the hope that it will be useful, but
** WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
** General Public License for more details.
**
** The GNU General Public License is available from http://www.gnu.org/ or
** the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
** MA 02111-1307, USA.
**
** For information on splint: info@splint.org
** To report a bug: splint-bug@splint.org
** For more information: http://www.splint.org
*/
/*
** source.c
**
** Interface to source file abstraction
**
** NOTE: This module is almost identical to the one for LCL. The
** only difference is that a couple of source lines have been
** commented out.
**
** This module has too many dependencies to be in the common
** source area. Any of the solutions that would allow this
** module to be common had its own set of compromises. It
** seemed best and most straightforward to just keep separte
** copies for LSL and LCL. We should examine this again if we
** ever reorganize the module structure.
**
** AUTHORS:
**
** Steve Garland,
** Massachusetts Institute of Technology
** Joe Wild, Technical Languages and Environments, DECspec project
*/
# include "splintMacros.nf"
# include "basic.h"
# include "osd.h"
extern bool
inputStream_close (inputStream s)
{
llassert (inputStream_isDefined (s));
if (s->file != NULL)
{
check (fileTable_closeFile (context_fileTable (), s->file));
s->file = NULL;
return TRUE;
}
return FALSE;
}
extern void
inputStream_free (/*@null@*/ /*@only@*/ inputStream s)
{
if (inputStream_isDefined (s))
{
cstring_free (s->name);
cstring_free (s->stringSource);
sfree (s);
}
}
extern /*@only@*/ inputStream
inputStream_create (cstring name, cstring suffix, bool echo)
{
char *ps;
inputStream s = (inputStream) dmalloc (sizeof (*s));
s->name = name;
s->file = NULL;
/*@access cstring@*/
llassert (cstring_isDefined (s->name));
ps = strrchr (s->name, CONNECTCHAR);
if (ps == NULL)
{
ps = s->name;
}
/*@noaccess cstring@*/
if (strchr (ps, '.') == NULL)
{
s->name = cstring_concatFree1 (s->name, suffix);
}
s->name = fileLib_cleanName (s->name);
s->lineNo = 0;
s->charNo = 0;
s->curLine = NULL;
s->echo = echo;
s->fromString = FALSE;
s->stringSource = NULL;
s->stringSourceTail = NULL;
s->buffer[0] = '\0';
return s;
}
extern /*@only@*/ inputStream
inputStream_fromString (cstring name, cstring str)
{
inputStream s = (inputStream) dmalloc (sizeof (*s));
s->name = cstring_copy (name);
s->stringSource = cstring_copy (str);
s->stringSourceTail = s->stringSource;
s->file = 0;
s->echo = FALSE;
s->fromString = TRUE;
s->lineNo = 0;
s->charNo = 0;
s->curLine = NULL;
s->buffer[0] = '\0';
return s;
}
extern int inputStream_nextChar (inputStream s)
{
int res;
llassert (inputStream_isDefined (s));
res = inputStream_peekChar (s);
if (res != EOF)
{
if (res == (int) '\n')
{
s->curLine = NULL;
s->charNo = 0;
incLine ();
}
else
{
s->charNo++;
incColumn ();
}
}
DPRINTF (("Next char: %c [%d]", (char) res, res));
return res;
}
extern int inputStream_peekNChar (inputStream s, int n)
/* Doesn't work across lines! */
{
llassert (inputStream_isDefined (s));
llassert (s->curLine != NULL);
llassert (s->charNo + n < strlen (s->curLine));
return ((int) s->curLine [s->charNo + n]);
}
extern int inputStream_peekChar (inputStream s)
{
llassert (inputStream_isDefined (s));
if (s->curLine == NULL)
{
char *cur;
s->curLine = NULL;
cur = inputStream_nextLine (s);
s->curLine = cur; /* split this to avoid possible undefined behavior */
s->charNo = 0;
}
if (s->curLine == NULL)
{
return EOF;
}
llassert (s->charNo <= strlen (s->curLine));
if (s->curLine[s->charNo] == '\0')
{
return (int) '\n';
}
return ((int) s->curLine [s->charNo]);
}
extern /*@dependent@*/ /*@null@*/
char *inputStream_nextLine (inputStream s)
{
char *currentLine;
size_t len;
llassert (inputStream_isDefined (s));
llassert (s->curLine == NULL);
s->charNo = 0;
if (s->fromString)
{
if (cstring_isEmpty (s->stringSourceTail))
{
currentLine = 0;
}
else
{
/*@access cstring@*/
char *c = strchr (s->stringSourceTail, '\n');
/* in case line is terminated not by newline */
if (c == 0)
{
c = strchr (s->stringSourceTail, '\0');
}
len = size_fromInt (c - s->stringSourceTail + 1);
if (len > size_fromInt (STUBMAXRECORDSIZE - 2))
{
len = size_fromInt (STUBMAXRECORDSIZE - 2);
}
currentLine = &(s->buffer)[0];
strncpy (currentLine, s->stringSourceTail, len);
currentLine[len] = '\0';
s->stringSourceTail += len;
/*@noaccess cstring@*/
}
}
else
{
llassert (s->file != NULL);
currentLine = fgets (&(s->buffer)[0], STUBMAXRECORDSIZE, s->file);
}
if (currentLine == 0)
{
strcpy (s->buffer, "*** End of File ***");
}
else
{
s->lineNo++;
len = strlen (currentLine) - 1;
if (s->buffer[len] == '\n')
{
s->buffer[len] = '\0';
}
else
{
if (len >= size_fromInt (STUBMAXRECORDSIZE - 2))
{
lldiagmsg (message ("Input line too long: %s",
cstring_fromChars (currentLine)));
}
}
}
/* if (s->echo) slo_echoLine (currentLine); only needed in LCL */
return currentLine;
}
extern bool
inputStream_open (inputStream s)
{
llassert (inputStream_isDefined (s));
if (s->fromString)
{
/* not an error: tail is dependent */
s->stringSourceTail = s->stringSource;
return TRUE;
}
DPRINTF (("Opening: %s", s->name));
s->file = fileTable_openReadFile (context_fileTable (), s->name);
return (s->file != 0 || s->fromString);
}
/*
** requires
** path != NULL \and
** s != NULL \and
** *s.name == filename (*s.name) || filetype (*s.name)
** *s.name consists of a file name and type only ("<filename>.<type>)
** No path name is included
**
** ensures
** if filefound (*path, *s) then
** result = true \and *s.name = filespec_where_file_found (*path, *s)
** else
** result = false
*/
extern bool inputStream_getPath (cstring path, inputStream s)
{
cstring returnPath;
filestatus status; /* return status of osd_getEnvPath.*/
bool rVal; /* return value of this procedure. */
llassert (cstring_isDefined (path));
llassert (inputStream_isDefined (s));
llassert (cstring_isDefined (s->name));
status = osd_getPath (path, s->name, &returnPath);
if (status == OSD_FILEFOUND)
{ /* Should be majority of cases. */
rVal = TRUE;
cstring_free (s->name);
s->name = fileLib_cleanName (returnPath);
}
else if (status == OSD_FILENOTFOUND)
{
rVal = FALSE;
}
else if (status == OSD_PATHTOOLONG)
{
rVal = FALSE;
/* Directory and filename are too long. Report error. */
llbuglit ("soure_getPath: Filename plus directory from search path too long");
}
else
{
rVal = FALSE;
llbuglit ("inputStream_getPath: invalid return status");
}
return rVal;
}
/*:open:*/ FILE *inputStream_getFile (inputStream s)
{
llassert (inputStream_isDefined (s));
llassert (s->file != NULL);
return s->file;
}
cstring inputStream_fileName (inputStream s)
{
llassert (inputStream_isDefined (s));
return s->name;
}
bool inputStream_isOpen (inputStream s)
{
return (inputStream_isDefined (s) && (s->file != 0 || s->fromString));
}
int inputStream_thisLineNumber (inputStream s)
{
llassert (inputStream_isDefined (s));
return s->lineNo;
}
syntax highlighted by Code2HTML, v. 0.9.1