mirror of
https://github.com/byteworksinc/ORCA-C.git
synced 2025-01-10 12:30:03 +00:00
6c1ccc5c0d
commit 4265329097538640e9e21202f1b141bcd42a44f3 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 21:45:32 2018 -0400 indent to match standard indent. commit 783518fbeb01d2df43ef2083d3341004c05e4e2e Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 20:21:15 2018 -0400 clean up the typenames commit 29b627ecf5ca9b8a143761f85a1807a6ca35ddd9 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 20:18:04 2018 -0400 enable feature_hh, warn about %n with non-int modifier. commit fc4ac8129e3772c4eda36658e344ec475938369c Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 15:13:47 2018 -0400 warn thar %lc, %ls, etc are unsupported. commit 7e6b433ba0552f7e52f0f034d398e9195c764326 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 13:36:25 2018 -0400 warn about hh/ll modifier (if not supported) commit 1943c9979d0013f9f38045ec04a962fbf0269f31 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Fri Mar 23 11:42:41 2018 -0400 use error facilities for format errors. commit 7811168f56dca1387055574ba8d32638da2fad96 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Thu Mar 22 15:34:21 2018 -0400 add feature flags to disable c99 enhancements until orca lib is updated. commit c2149cc5953155cfc3c3b4d0483cd25fb946b055 Author: Kelvin Sherlock <ksherlock@gmail.com> Date: Thu Mar 22 08:59:10 2018 -0400 Add printf/scanf format checking [WIP] This parses out the xprintf / xscanf format string and compares it with the function arguments. enabled via #pragma lint 16.
835 lines
21 KiB
ObjectPascal
835 lines
21 KiB
ObjectPascal
{$optimize 7}
|
|
{---------------------------------------------------------------}
|
|
{ }
|
|
{ Printf }
|
|
{ }
|
|
{ Analyzes printf/scanf format and arguments to check for }
|
|
{ potential problems. }
|
|
{ Called from FunctionParms (Expression.pas) }
|
|
{ Enabled via #pragma lint 16 }
|
|
{---------------------------------------------------------------}
|
|
|
|
unit Printf;
|
|
|
|
interface
|
|
|
|
{$LibPrefix '0/obj/'}
|
|
|
|
uses CCommon;
|
|
|
|
{$segment 'PRINTF'}
|
|
|
|
type
|
|
|
|
fmtArgPtr = ^fmtArgRecord;
|
|
|
|
fmtArgRecord = record
|
|
next: fmtArgPtr;
|
|
ty: typePtr;
|
|
tk: tokenPtr;
|
|
end;
|
|
|
|
{
|
|
format arg1: printf
|
|
format arg2: fprintf, sprintf, asprintf, dprintf
|
|
format arg3: snprintf
|
|
|
|
format arg1: scanf
|
|
format arg2: fscanf, sscanf
|
|
}
|
|
fmt_type = (fmt_none, fmt_printf1, fmt_printf2, fmt_printf3, fmt_scanf1, fmt_scanf2);
|
|
|
|
|
|
function FormatClassify(fname: stringPtr): fmt_type;
|
|
|
|
procedure FormatCheck(fmt: fmt_type; args: fmtArgPtr);
|
|
|
|
|
|
implementation
|
|
|
|
const
|
|
feature_hh = true;
|
|
feature_ll = false;
|
|
feature_s_long = false;
|
|
feature_n_size = false;
|
|
|
|
type
|
|
length_modifier = (default, h, hh, l, ll, j, z, t, ld);
|
|
|
|
state_enum = (st_text, st_flag, st_width,
|
|
st_precision_dot, st_precision, st_precision_number,
|
|
st_length, st_length_h, st_length_l, st_format,
|
|
{ scanf }
|
|
st_suppress, st_set, st_set_1, st_set_2,
|
|
st_error);
|
|
|
|
types = set of baseTypeEnum;
|
|
|
|
procedure Error (err: integer); extern; {in scanner.pas}
|
|
|
|
|
|
|
|
function FormatClassify {fname: stringPtr): fmt_type};
|
|
{
|
|
Check if a function name is printf/scanf. Caller must check if
|
|
it otherwise matches (variadic, direct call)
|
|
}
|
|
|
|
var
|
|
l: integer;
|
|
|
|
begin {FormatClassify}
|
|
|
|
FormatClassify := fmt_none;
|
|
|
|
l := length(fname^);
|
|
if (l >= 5) and (l <= 8) then case fname^[1] of
|
|
'a': if fname^ = 'asprintf' then FormatClassify := fmt_printf2;
|
|
'd': if fname^ = 'dprintf' then FormatClassify := fmt_printf2;
|
|
'p': if fname^ = 'printf' then FormatClassify := fmt_printf1;
|
|
'f':
|
|
if fname^ = 'fprintf' then FormatClassify := fmt_printf2
|
|
else if fname^ = 'fscanf' then FormatClassify := fmt_scanf2;
|
|
's':
|
|
if fname^ = 'scanf' then FormatClassify := fmt_scanf1
|
|
else if fname^ = 'snprintf' then FormatClassify := fmt_printf3
|
|
else if fname^ = 'sprintf' then FormatClassify := fmt_printf2
|
|
else if fname^ = 'sscanf' then FormatClassify := fmt_scanf2;
|
|
otherwise: ;
|
|
end; {case}
|
|
end; {FormatClassify}
|
|
|
|
|
|
procedure FormatCheck{fmt: fmt_type; args: fmtArgPtr};
|
|
|
|
var
|
|
head: fmtArgPtr;
|
|
s: longstringPtr;
|
|
state: state_enum;
|
|
has_length: length_modifier;
|
|
error_count: integer;
|
|
expected: integer;
|
|
offset: integer;
|
|
|
|
|
|
number_set : set of char;
|
|
flag_set : set of char;
|
|
length_set : set of char;
|
|
format_set : set of char;
|
|
|
|
|
|
|
|
procedure Warning(msg: stringPtr);
|
|
{
|
|
Pretty Print a warning.
|
|
offset is the location of the current % character within s.
|
|
}
|
|
var
|
|
i: integer;
|
|
c: char;
|
|
|
|
begin {Warning}
|
|
if error_count = 0 then begin
|
|
Error(124);
|
|
if s <> nil then begin
|
|
Write(' "');
|
|
for i := 1 to s^.length do begin
|
|
c := s^.str[i];
|
|
if (c = '"') or (ord(c) < $20) or (ord(c) > $7f) then c := '.';
|
|
Write(c);
|
|
end; {for}
|
|
WriteLn('"');
|
|
end; {if}
|
|
end; {if}
|
|
error_count := error_count + 1;
|
|
Write(' ');
|
|
if offset <> 0 then begin
|
|
for i := 1 to offset do Write(' ');
|
|
Write('^ ');
|
|
end; {if}
|
|
Write('Warning: ');
|
|
WriteLn(msg^);
|
|
end; {Warning}
|
|
|
|
procedure WarningConversionChar(c: char);
|
|
{ Warn that a conversion character is invalid, eg %z }
|
|
var
|
|
msg: stringPtr;
|
|
|
|
begin {WarningConversionChar}
|
|
if (ord(c) >= $20) and (ord(c) <= $7f) then begin
|
|
new(msg);
|
|
msg^ := concat('unknown conversion type character ''', c, ''' in format.');
|
|
Warning(msg);
|
|
dispose(msg);
|
|
end {if}
|
|
else Warning(@'unknown conversion type character in format.');
|
|
end; {WarningConversionChar}
|
|
|
|
procedure WarningExtraArgs(i: integer);
|
|
{ Warn that too many arguments were provided }
|
|
var
|
|
msg: stringPtr;
|
|
begin {WarningExtraArgs}
|
|
new(msg);
|
|
msg^ := concat('extra arguments provided (', cnvis(i), ' expected).');
|
|
Warning(msg);
|
|
dispose(msg);
|
|
end; {WarningExtraArgs}
|
|
|
|
|
|
|
|
function popType: typePtr;
|
|
{ Return the token type and advance the linked list. }
|
|
begin {popType}
|
|
expected := expected + 1;
|
|
popType := nil;
|
|
if args <> nil then begin
|
|
popType := args^.ty;
|
|
args := args^.next;
|
|
end; {if}
|
|
end; {popType}
|
|
|
|
|
|
procedure expect_long;
|
|
{ Verify the current argument is a long int.}
|
|
var
|
|
ty: typePtr;
|
|
|
|
begin {expect_long}
|
|
ty := popType;
|
|
if ty <> nil then begin
|
|
if (ty^.kind <> scalarType) or (not (ty^.baseType in [cgLong, cgULong])) then begin
|
|
Warning(@'expected long int.');
|
|
end; {if}
|
|
end {if}
|
|
else begin
|
|
Warning(@'argument missing; expected long int');
|
|
end; {else}
|
|
end; {expect_long}
|
|
|
|
procedure expect_int;
|
|
var
|
|
ty: typePtr;
|
|
|
|
begin {expect_int}
|
|
ty := popType;
|
|
if ty <> nil then begin
|
|
if (ty^.kind <> scalarType) or
|
|
not (ty^.baseType in [cgWord, cgUWord, cgByte, cgUByte]) then begin
|
|
Warning(@'expected int.');
|
|
end; {if}
|
|
end {if}
|
|
else begin
|
|
Warning(@'argument missing; expected int.');
|
|
end; {else}
|
|
end; {expect_int}
|
|
|
|
|
|
procedure expect_char;
|
|
var
|
|
ty: typePtr;
|
|
|
|
begin {expect_char}
|
|
ty := popType;
|
|
if ty <> nil then begin
|
|
if (ty^.kind <> scalarType) or
|
|
not (ty^.baseType in [cgWord, cgUWord, cgByte, cgUByte]) then begin
|
|
Warning(@'expected char.');
|
|
end; {if}
|
|
end {if}
|
|
else begin
|
|
Warning(@'argument missing; expected char.');
|
|
end; {else}
|
|
end; {expect_char}
|
|
|
|
procedure expect_extended;
|
|
{ Verify the current argument is an extended*. }
|
|
{ * or float or double since they're all passed as extended }
|
|
var
|
|
ty: typePtr;
|
|
|
|
begin {expect_extended}
|
|
ty := popType;
|
|
if ty <> nil then begin
|
|
if (ty^.kind <> scalarType) or
|
|
not (ty^.baseType in [cgExtended, cgReal, cgDouble]) then begin
|
|
Warning(@'expected extended.');
|
|
end; {if}
|
|
end {if}
|
|
else begin
|
|
Warning(@'argument missing; expected extended.');
|
|
end; {else}
|
|
end; {expect_extended}
|
|
|
|
procedure expect_pointer;
|
|
{ Verify the current argument is a pointer of some sort. }
|
|
var
|
|
ty: typePtr;
|
|
|
|
begin {expect_pointer}
|
|
ty := popType;
|
|
if ty <> nil then begin
|
|
if (ty^.kind <> pointerType) then begin
|
|
Warning(@'expected pointer.');
|
|
end; {if}
|
|
end {if}
|
|
else begin
|
|
Warning(@'argument missing; expected pointer.');
|
|
end; {else}
|
|
end; {expect_pointer}
|
|
|
|
procedure expect_pointer_to(expected: types; name: stringPtr);
|
|
{ Verify the current argument is a pointer to the expected set.}
|
|
var
|
|
ty: typePtr;
|
|
baseTy: typePtr;
|
|
ok: boolean;
|
|
|
|
procedure error(prefix: stringPtr);
|
|
var
|
|
msg: stringPtr;
|
|
begin
|
|
new(msg);
|
|
msg^ := concat(prefix^, name^, '.');
|
|
Warning(msg);
|
|
dispose(msg);
|
|
end; {error}
|
|
|
|
begin {expect_pointer_to}
|
|
ok := false;
|
|
ty := popType;
|
|
baseTy := nil;
|
|
|
|
|
|
if ty <> nil then
|
|
if (ty^.kind = pointerType) or (ty^.kind = arrayType) then begin
|
|
baseTy := ty^.pType;
|
|
if (baseTy <> nil)
|
|
and (baseTy^.kind = scalarType)
|
|
and (baseTy^.baseType in expected)
|
|
then ok := true;
|
|
end; {if}
|
|
|
|
if not ok then begin
|
|
if ty = nil then
|
|
error(@'argument missing; expected pointer to ')
|
|
else error(@'expected pointer to ');
|
|
end; {if}
|
|
|
|
end; {expect_pointer_to}
|
|
|
|
procedure do_length(c: char);
|
|
{ helper to process the length modifier }
|
|
begin {do_length}
|
|
state := st_format;
|
|
case c of
|
|
'h': begin
|
|
has_length := h;
|
|
state := st_length_h;
|
|
end;
|
|
'l': begin
|
|
has_length := l;
|
|
state := st_length_l;
|
|
end;
|
|
'j': has_length := j;
|
|
'z': has_length := z;
|
|
't': has_length := t;
|
|
'L': has_length := ld;
|
|
end; {case}
|
|
end; {do_length}
|
|
|
|
|
|
|
|
procedure FormatScanf;
|
|
{ Check the scanf string and arguments. }
|
|
|
|
label 1;
|
|
|
|
var
|
|
i: integer;
|
|
c: char;
|
|
has_suppress: boolean;
|
|
|
|
|
|
|
|
procedure do_scanf_format;
|
|
|
|
{ check an individual scanf argument. }
|
|
|
|
{
|
|
(current) ORCALib limitations, wrt size modifiers:
|
|
|
|
- ignored for string types
|
|
- hh not supported
|
|
- L not supported
|
|
- ignored for 'n'
|
|
}
|
|
var
|
|
expected: types;
|
|
name: stringPtr;
|
|
|
|
begin {do_scanf_format}
|
|
|
|
name := nil;
|
|
|
|
state := st_text;
|
|
if c in format_set then begin
|
|
|
|
case c of
|
|
|
|
'%': has_suppress := true;
|
|
|
|
'c', 'b', 's', '[' : begin
|
|
{ %ls, etc is a wchar_t *}
|
|
|
|
expected := [cgByte, cgUByte];
|
|
name := @'char';
|
|
|
|
if has_length = l then begin
|
|
expected := [cgWord, cgUWord];
|
|
name := @'wchar';
|
|
|
|
if not feature_s_long then
|
|
Warning(@'%ls not currently supported');
|
|
|
|
end; {if}
|
|
|
|
if c = '[' then state := st_set_1;
|
|
end;
|
|
|
|
'd', 'i', 'u', 'o', 'x', 'X': begin
|
|
case has_length of
|
|
hh: begin
|
|
expected := [cgByte, cgUByte];
|
|
name := @'char';
|
|
end;
|
|
l, ll, j, z, t: begin
|
|
expected := [cgLong, cgULong];
|
|
name := @'long';
|
|
end;
|
|
otherwise: begin
|
|
expected := [cgWord, cgUWord];
|
|
name := @'int';
|
|
end;
|
|
end; {case}
|
|
end;
|
|
|
|
'n': begin
|
|
{ ORCALib always treats n as int * }
|
|
{ n.b. - *n is undefined; orcalib pops a parm but doesn't store.}
|
|
{ C99 - support for length modifiers }
|
|
if has_suppress then Warning(@'*n is undefined.');
|
|
has_suppress := false;
|
|
|
|
if (not feature_n_size) and (has_length <> default) then
|
|
Warning(@'size modifier for %n not currently supported.');
|
|
|
|
case has_length of
|
|
hh: begin
|
|
expected := [cgByte, cgUByte];
|
|
name := @'char';
|
|
end;
|
|
l, ll, j, z, t: begin
|
|
expected := [cgLong, cgULong];
|
|
name := @'long';
|
|
end;
|
|
otherwise: begin
|
|
expected := [cgWord, cgUWord];
|
|
name := @'int';
|
|
end;
|
|
end; {case}
|
|
end;
|
|
'p': begin
|
|
if not has_suppress then expect_pointer;
|
|
has_suppress := true;
|
|
end;
|
|
'a', 'A', 'f', 'F', 'g', 'G', 'e', 'E': begin
|
|
|
|
case has_length of
|
|
ld: begin
|
|
expected := [cgExtended];
|
|
name := @'long double';
|
|
end;
|
|
l: begin
|
|
expected := [cgDouble];
|
|
name := @'double';
|
|
end;
|
|
otherwise: begin
|
|
expected := [cgReal];
|
|
name := @'float';
|
|
end;
|
|
end; {case}
|
|
end;
|
|
end; { case }
|
|
|
|
if not has_suppress then begin
|
|
expect_pointer_to(expected, name);
|
|
end; {if}
|
|
|
|
end {if}
|
|
else WarningConversionChar(c);
|
|
|
|
|
|
end; {do_scanf_format}
|
|
|
|
|
|
|
|
begin {FormatScanf}
|
|
|
|
{
|
|
'%'
|
|
'*'? - assignment suppression
|
|
\d* - maximum field width
|
|
(h|hh|l|ll|j|z|t|L)? - length modifier
|
|
[%bcsdiuoxXnaAeEfFgGp] | set - format
|
|
|
|
set: '[[' [^]]* ']'
|
|
set: '[^[' [^]]* ']'
|
|
set: '[' [^]]+ ']'
|
|
|
|
}
|
|
state := st_text;
|
|
expected := 0;
|
|
offset := 0;
|
|
|
|
number_set := ['0' .. '9'];
|
|
length_set := ['h', 'l', 'j', 't', 'z', 'L'];
|
|
flag_set := ['#', '0', '-', '+', ' '];
|
|
format_set := ['%', '[', 'b', 'c', 's', 'd', 'i', 'o', 'x', 'X', 'u',
|
|
'f', 'F', 'e', 'E', 'a', 'A', 'g', 'G', 'n', 'p'];
|
|
|
|
|
|
for i := 1 to s^.length do begin
|
|
c := s^.str[i];
|
|
case state of
|
|
st_text: if c = '%' then begin
|
|
state := st_suppress;
|
|
offset := i;
|
|
has_length := default;
|
|
has_suppress := false;
|
|
end; {if}
|
|
|
|
st_suppress: { suppress? width? length? format }
|
|
if c = '*' then begin
|
|
state := st_width;
|
|
has_suppress := true;
|
|
end {if}
|
|
else if c in number_set then state := st_width
|
|
else if c in length_set then do_length(c)
|
|
else do_scanf_format;
|
|
|
|
st_width: {width? length? format }
|
|
if c in number_set then state := st_width
|
|
else if c in length_set then do_length(c)
|
|
else do_scanf_format;
|
|
|
|
st_length_h: { h? format }
|
|
if c = 'h' then begin
|
|
has_length := hh;
|
|
state := st_format;
|
|
if not feature_hh then
|
|
Warning(@'hh modifier not currently supported');
|
|
end {if}
|
|
else do_scanf_format;
|
|
|
|
st_length_l: { l? format }
|
|
if c = 'l' then begin
|
|
has_length := ll;
|
|
state := st_format;
|
|
if not feature_ll then
|
|
Warning(@'ll modifier not currently supported');
|
|
end {if}
|
|
else do_scanf_format;
|
|
|
|
st_format: { format }
|
|
do_scanf_format;
|
|
|
|
{ first char of a [set]. ']' does not end the set. }
|
|
st_set_1:
|
|
if c = '^' then state := st_set_2
|
|
else state := st_set;
|
|
|
|
st_set_2:
|
|
state := st_set;
|
|
|
|
st_set:
|
|
if c = ']' then state := st_text;
|
|
|
|
st_error: goto 1;
|
|
end; { case }
|
|
end; { for }
|
|
|
|
if state <> st_text then
|
|
Warning(@'incomplete format specifier.');
|
|
|
|
if args <> nil then begin
|
|
offset := 0;
|
|
WarningExtraArgs(expected);
|
|
end;
|
|
|
|
1:
|
|
|
|
end; {FormatScanf}
|
|
|
|
|
|
|
|
procedure FormatPrintf;
|
|
{ Check the printf string and arguments. }
|
|
|
|
label 1;
|
|
|
|
var
|
|
|
|
i : integer;
|
|
c : char;
|
|
|
|
has_flag : boolean;
|
|
has_width: boolean;
|
|
has_precision : boolean;
|
|
|
|
procedure do_printf_format;
|
|
{ check an individual printf argument. }
|
|
|
|
begin {do_printf_format}
|
|
state := st_text;
|
|
if c in format_set then begin
|
|
case c of
|
|
'p': expect_pointer;
|
|
|
|
{ %b: orca-specific - pascal string }
|
|
'b', 's':
|
|
if has_length = l then begin
|
|
if not feature_s_long then
|
|
Warning(@'%ls not currently supported.');
|
|
|
|
expect_pointer_to([cgWord, cgUWord], @'wchar')
|
|
end {if}
|
|
else expect_pointer_to([cgByte, cgUByte], @'char');
|
|
|
|
'n': begin
|
|
|
|
if (not feature_n_size) and (has_length <> default) then
|
|
Warning(@'size modifier for %n not currently supported.');
|
|
|
|
case has_length of
|
|
hh:
|
|
expect_pointer_to([cgByte, cgUByte], @'char');
|
|
|
|
l, ll, j, z, t:
|
|
expect_pointer_to([cgLong, cgULong], @'long');
|
|
|
|
otherwise:
|
|
expect_pointer_to([cgWord, cgUWord], @'int');
|
|
end; {case}
|
|
|
|
end;
|
|
|
|
'c':
|
|
if has_length = l then begin
|
|
if not feature_s_long then Warning(@'%lc not currently supported');
|
|
expect_int;
|
|
end
|
|
else begin
|
|
expect_char;
|
|
end;
|
|
|
|
{ chars are passed as ints so %hhx can be ignored here. }
|
|
'd', 'i', 'o', 'x', 'X', 'u':
|
|
if has_length in [l, ll, j, z, t] then begin
|
|
expect_long;
|
|
end
|
|
else begin
|
|
expect_int;
|
|
end;
|
|
|
|
'f', 'F', 'e', 'E', 'a', 'A', 'g', 'G':
|
|
expect_extended;
|
|
'%': ;
|
|
end; {case}
|
|
end {if}
|
|
else WarningConversionChar(c);
|
|
|
|
|
|
end; {do_printf_format}
|
|
|
|
begin {FormatPrintf}
|
|
|
|
state := st_text;
|
|
expected := 0;
|
|
offset := 0;
|
|
|
|
number_set := ['0' .. '9'];
|
|
length_set := ['h', 'l', 'j', 't', 'z', 'L'];
|
|
flag_set := ['#', '0', '-', '+', ' '];
|
|
format_set := ['%', 'b', 'c', 's', 'd', 'i', 'o', 'x', 'X', 'u',
|
|
'f', 'F', 'e', 'E', 'a', 'A', 'g', 'G', 'n', 'p'];
|
|
|
|
for i := 1 to s^.length do begin
|
|
c := s^.str[i];
|
|
case state of
|
|
st_text:
|
|
if c = '%' then begin
|
|
state := st_flag;
|
|
offset := i;
|
|
has_length := default;
|
|
has_flag := false;
|
|
has_width := false;
|
|
has_precision := false;
|
|
end;
|
|
|
|
st_flag: { flags* width? precision? length? format }
|
|
if c in flag_set then begin
|
|
state := st_flag;
|
|
has_flag := true;
|
|
end
|
|
else if c in number_set then begin
|
|
state := st_width;
|
|
has_width := true;
|
|
end
|
|
else if c = '*' then begin
|
|
{ * for the width }
|
|
has_width := true;
|
|
expect_int;
|
|
state := st_precision;
|
|
end
|
|
else if c = '.' then state := st_precision_dot
|
|
else if c in length_set then do_length(c)
|
|
else do_printf_format;
|
|
|
|
st_width: { width? precision? length? format }
|
|
if c in number_set then state := st_width
|
|
else if c = '.' then state := st_precision_dot
|
|
else if c in length_set then do_length(c)
|
|
else do_printf_format;
|
|
|
|
st_precision: { (. precision)? length? format }
|
|
if c = '.' then state := st_precision_dot
|
|
else if c in length_set then do_length(c)
|
|
else do_printf_format;
|
|
|
|
|
|
st_precision_dot: begin { * | [0-9]+ }
|
|
has_precision := true;
|
|
if c = '*' then begin
|
|
expect_int;
|
|
state := st_length;
|
|
end
|
|
else if c in number_set then state := st_precision_number
|
|
else state := st_error;
|
|
end;
|
|
|
|
st_precision_number: { [0-9]* length? format }
|
|
if c in number_set then state := st_precision_number
|
|
else if c in length_set then do_length(c)
|
|
else do_printf_format;
|
|
|
|
st_length: { length? format }
|
|
if c in length_set then do_length(c)
|
|
else do_printf_format;
|
|
|
|
st_length_h: { h? format }
|
|
if c = 'h' then begin
|
|
has_length := hh;
|
|
state := st_format;
|
|
if not feature_hh then
|
|
Warning(@'hh modifier not currently supported');
|
|
end
|
|
else do_printf_format;
|
|
|
|
st_length_l: { l? format}
|
|
if c = 'l' then begin
|
|
has_length := ll;
|
|
state := st_format;
|
|
if not feature_ll then
|
|
Warning(@'ll modifier not currently supported');
|
|
end
|
|
else do_printf_format;
|
|
|
|
st_format: do_printf_format;
|
|
|
|
st_error: { error }
|
|
goto 1;
|
|
|
|
end; { case }
|
|
end; { for i }
|
|
|
|
if state <> st_text then
|
|
Warning(@'incomplete format specifier.');
|
|
|
|
if args <> nil then begin
|
|
offset := 0;
|
|
WarningExtraArgs(expected);
|
|
end;
|
|
1:
|
|
|
|
end; {FormatPrintf}
|
|
|
|
|
|
|
|
|
|
|
|
function get_format_string(pos: integer): longstringPtr;
|
|
{ get the format string from the pos'th argument. }
|
|
var
|
|
tk: tokenPtr;
|
|
|
|
begin {get_format_string}
|
|
get_format_string := nil;
|
|
|
|
while (args <> nil) and (pos > 1) do begin
|
|
args := args^.next;
|
|
pos := pos - 1;
|
|
end; {while}
|
|
|
|
if (pos = 1) and (args <> nil) then begin
|
|
tk := args^.tk;
|
|
args := args^.next;
|
|
|
|
if (tk <> nil) and (tk^.token.kind = stringconst) then
|
|
get_format_string := tk^.token.sval
|
|
else
|
|
Error(125);
|
|
end; {if}
|
|
{ no format string -> Error(85) }
|
|
end; {get_format_string}
|
|
|
|
|
|
|
|
begin {FormatCheck}
|
|
|
|
head := args;
|
|
error_count := 0;
|
|
offset := 0;
|
|
|
|
case fmt of
|
|
fmt_printf1, fmt_scanf1:
|
|
s := get_format_string(1);
|
|
|
|
fmt_printf2, fmt_scanf2:
|
|
s := get_format_string(2);
|
|
|
|
fmt_printf3:
|
|
s := get_format_string(3);
|
|
|
|
otherwise: s := nil;
|
|
end; {case}
|
|
|
|
if (s <> nil) then case fmt of
|
|
fmt_printf1, fmt_printf2, fmt_printf3:
|
|
FormatPrintf;
|
|
|
|
fmt_scanf1, fmt_scanf2:
|
|
FormatScanf;
|
|
end; {case}
|
|
|
|
{ clean up linked list }
|
|
while head <> nil do begin
|
|
args := head^.next;
|
|
dispose(head);
|
|
head := args;
|
|
end;
|
|
end; {FormatCheck}
|
|
|
|
end.
|