File: pgsymtab.c

package info (click to toggle)
ftnchek 3.1.2-2
  • links: PTS
  • area: main
  • in suites: woody
  • size: 6,436 kB
  • ctags: 5,393
  • sloc: ansic: 24,609; fortran: 5,565; yacc: 3,682; sh: 2,518; makefile: 772; lisp: 264; f90: 94; perl: 76
file content (441 lines) | stat: -rw-r--r-- 11,376 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
/* $Id: pgsymtab.c,v 1.39 2001/10/07 22:59:51 moniot Exp $

	 Warning message routines for printing of global symbol table info

*/

/*

Copyright (c) 2001 by Robert K. Moniot.

Permission is hereby granted, free of charge, to any person
obtaining a copy of this software and associated documentation
files (the "Software"), to deal in the Software without
restriction, including without limitation the rights to use,
copy, modify, merge, publish, distribute, sublicense, and/or
sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following
conditions:

The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the
Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

Acknowledgement: the above permission notice is what is known
as the "MIT License."
*/

/*

    Shared functions defined:
	argcmp_error_head	prints intro to argument mismatch
	comcmp_error_head	prints intro to common mismatch
	arg_error_report	Follow-on message about an argument mismatch
	sub_error_report	Error message line about one subprogram
				   invocation
	com_error_report	Error message line about one common block
				   declaration
	comvar_error_report	Error message line about one common var
				   mismatch
	sort_gsymbols		sort a list of Gsymtab ptrs alphabetically

*/

#include <stdio.h>
#include <ctype.h>
#include <string.h>
#include "ftnchek.h"
#define PGSYMTAB
#include "symtab.h"
#include "pgsymtab.h"


			/* Private functions defined: */
PROTO(PRIVATE void arg_error_locate,( ArgListHeader *alh ));
PROTO(PRIVATE int cmp_error_head,(const char *name, const char *tag, const char *filename,
				  LINENO_t lineno, const char *msg ));
PROTO(PRIVATE void com_error_locate,( ComListHeader *clh ));
PROTO(PRIVATE void error_report,( const char *module_name,
				  const char *filename, LINENO_t lineno,
				  const char *topfile, LINENO_t top_lineno,
	     int i, const char *item_tag, const char *item_name, const char *msg ));
PROTO(PRIVATE void module_locate, (const char *name));
PROTO(PRIVATE void novice_err_locate,( const char *filename, LINENO_t linenum ));
PROTO(PRIVATE void novice_inc_locate,( const char *filename, const char *topfile,
				       LINENO_t top_linenum ));
PROTO(PRIVATE void report_intro,( const char *filename, LINENO_t lineno,
				  const char *topfile, LINENO_t top_lineno ));

			/* Var used to control spaces betw message blocks */
PRIVATE int global_warning_count=0;


/****  Definitions of shared functions ****/



		/* Intro line of warning about subprogram argument mismatches.
		 */
int
#if HAVE_STDC
argcmp_error_head(const char *name, ArgListHeader *alh, const char *msg)
#else /* K&R style */
argcmp_error_head(name, alh, msg)
    char *name;
    ArgListHeader *alh;
    char *msg;
#endif /* HAVE_STDC */
{
    return cmp_error_head(name,"Subprogram",
			  alh->filename,
			  alh->line_num,
			  msg);
}

		/* Ditto for common block declaration mismatches.
		 */
int
#if HAVE_STDC
comcmp_error_head(const char *name, ComListHeader *clh, const char *msg)
#else /* K&R style */
comcmp_error_head(name, clh, msg)
    char *name;
    ComListHeader *clh;
    char *msg;
#endif /* HAVE_STDC */
{
    return cmp_error_head(name,"Common block",
			  clh->filename,
			  clh->line_num,
			  msg);
}

		/* Follow-on message about an argument mismatch */
void
#if HAVE_STDC
arg_error_report(ArgListHeader *alh, const char *argtype, int i, const char *msg)
#else /* K&R style */
arg_error_report(alh, argtype, i, msg)
    ArgListHeader *alh;
    char *argtype;
    int i;
    char *msg;
#endif /* HAVE_STDC */
{
    error_report(alh->module->name,
		 alh->filename,alh->line_num,alh->topfile,alh->top_line_num,
		 i,argtype,alh->arg_array[i].name,msg);
}

		/* Formats an error message line about one subprogram
		   invocation.
		*/
void
#if HAVE_STDC
sub_error_report(ArgListHeader *alh, const char *msg)
#else /* K&R style */
sub_error_report(alh, msg)
    ArgListHeader *alh;
    const char *msg;
#endif /* HAVE_STDC */
{
    report_intro(alh->filename,alh->line_num,alh->topfile,alh->top_line_num);
    msg_tail(msg);
    arg_error_locate(alh);
}



		/* Formats an error message line about one common block
		   declaration.
		*/
void
#if HAVE_STDC
com_error_report(ComListHeader *clh, const char *msg)
#else /* K&R style */
com_error_report(clh, msg)
    ComListHeader *clh;
    char *msg;
#endif /* HAVE_STDC */
{
    report_intro(clh->filename,clh->line_num,clh->topfile,clh->top_line_num);
    msg_tail(msg);
    com_error_locate(clh);
}

		/* Formats an error message line about one common var mismatch.
		*/
void
#if HAVE_STDC
comvar_error_report(ComListHeader *clh, int i, const char *msg)
#else /* K&R style */
comvar_error_report(clh, i, msg)
    ComListHeader *clh;
    int i;
    char *msg;
#endif /* HAVE_STDC */
{
    error_report(clh->module->name,
		 clh->filename,clh->line_num,clh->topfile,clh->top_line_num,
		 i,"Variable",clh->com_list_array[i].name,msg);
}


/**** Definitions of private functions ****/

PRIVATE int at_position_printed;


	/* Increment error count, and if it is 1, print header for arg or com
	   mismatch error messages.  If it is past limit, print "etc"
	   and return TRUE, otherwise return FALSE.
	   */
PRIVATE int
#if HAVE_STDC
cmp_error_head(const char *name, const char *tag,
	       const char *filename, LINENO_t lineno, const char *msg)
#else /* K&R style */
cmp_error_head(name, tag,
	       filename, lineno, msg)
    char *name;
    char *tag;
    char *filename;
    LINENO_t lineno;
    char *msg;
#endif /* HAVE_STDC */
{
		/* stop after limit: probably a cascade */
	if( CASCADE_LIMIT(cmp_error_count) ) {
	  (void)fprintf(list_fd,"\n etc...");
	  return TRUE;
	}

			/* (For expert mode, line number helps smart editors,
			   but in novice mode it looks silly to have a line
			   number for a mismatch involving two lines.)
			 */
	if(novice_help) {
	    filename = (char *)NULL;
	    lineno = NO_LINE_NUM;
	}

	if(cmp_error_count == 1) {
				/* If -noquiet mode, put a space between
				   successive warnings.
				 */
	    if( (global_warning_count != 0) && !quiet)
		(void)fprintf(list_fd,"\n");
	    global_warning_count++;
	    global_warning(filename,lineno,tag);
	    msg_tail(name);
	    msg_tail(msg);
	}
	else {			/* for "and at position n" on new line */
	    global_message(filename,lineno," and");
	}
	at_position_printed = FALSE;

	return FALSE;
}



PRIVATE void
#if HAVE_STDC
error_report(const char *module_name, const char *filename, LINENO_t lineno,
	     const char *topfile, LINENO_t top_lineno,
	     int i, const char *item_tag, const char *item_name, const char *msg)
#else /* K&R style */
error_report(module_name, filename, lineno,
	     topfile, top_lineno,
	     i, item_tag, item_name, msg)
    char *module_name;
    char *filename;
    LINENO_t lineno;
    char *topfile;
    LINENO_t top_lineno;
    int i;
    char *item_tag;
    char *item_name;
    char *msg;
#endif /* HAVE_STDC */
{
    if( ! at_position_printed ) {
	char posn[12+3*sizeof(int)+2];
	(void)sprintf(posn,"at position %d:",i+1);
	msg_tail(posn);
	at_position_printed = TRUE;
    }

    report_intro(filename,lineno,topfile,top_lineno);
    msg_tail(item_tag);

    msg_tail(item_name);
		/* Print module name, and for -novice mode, location info
		   that was suppressed before.
		 */
    module_locate(module_name);

    if( novice_help ) {
				/* Error location itself */
	novice_err_locate(filename,lineno);
				/* Location where included */
	novice_inc_locate(filename,topfile,top_lineno);
    }

    msg_tail(msg);

}


PRIVATE void
#if HAVE_STDC
report_intro(const char *filename, LINENO_t lineno,
	     const char *topfile, LINENO_t top_lineno)
#else /* K&R style */
report_intro(filename, lineno,
	     topfile, top_lineno)
    char *filename;
    LINENO_t lineno;
    char *topfile;
    LINENO_t top_lineno;
#endif /* HAVE_STDC */
{
			/* In expert mode, if error is in include file,
			   need to give info about it.
			 */
    if( ! novice_help && filename != topfile) {
	global_message(filename,lineno,"(location of error)");
	global_message(topfile,top_lineno,"(where included)  ");
    }
    else {  
	global_message(filename,lineno,"  ");
    }
}

	/* Gives module and in novice mode line, filename for error messages
	 */
PRIVATE void
#if HAVE_STDC
arg_error_locate(ArgListHeader *alh)
#else /* K&R style */
arg_error_locate(alh)
     ArgListHeader *alh;
#endif /* HAVE_STDC */
{
				/* Module (subprogram) containing the error.
				   This gets printed in both modes. */
    module_locate(alh->module->name);

    if( novice_help ) {
				/* Error location itself */
	novice_err_locate(alh->filename,alh->line_num);
				/* Location where included */
	novice_inc_locate(alh->filename,alh->topfile,alh->top_line_num);
    }
}

PRIVATE void
#if HAVE_STDC
com_error_locate(ComListHeader *clh)
#else /* K&R style */
com_error_locate(clh)
     ComListHeader *clh;
#endif /* HAVE_STDC */
{
				/* Module (subprogram) containing the error.
				   This gets printed in both modes. */
    module_locate(clh->module->name);

    if( novice_help ) {
				/* Error location itself */
	novice_err_locate(clh->filename,clh->line_num);
				/* Location where included */
	novice_inc_locate(clh->filename,clh->topfile,clh->top_line_num);
    }
}

PRIVATE void
#if HAVE_STDC
module_locate(const char *name)
#else /* K&R style */
module_locate(name)
    const char *name;
#endif /* HAVE_STDC */
{
    msg_tail("in module");
    msg_tail(name);
}

			/* Non-include part of location, novice mode */
PRIVATE void
#if HAVE_STDC
novice_err_locate(const char *filename, LINENO_t linenum)
#else /* K&R style */
novice_err_locate(filename,linenum)
     char *filename;
     LINENO_t linenum;
#endif /* HAVE_STDC */
{
    msg_tail("line");
    msg_tail(ulongtostr((unsigned long)linenum));
    msg_tail("file");
    msg_tail(filename);
}

			/* Include-file part of location, novice mode */
PRIVATE void
#if HAVE_STDC
novice_inc_locate(const char *filename, const char *topfile, LINENO_t top_linenum)
#else /* K&R style */
novice_inc_locate(filename,topfile,top_linenum)
    char *filename;
    char *topfile;
    LINENO_t top_linenum;
#endif /* HAVE_STDC */
{
    if(filename != topfile) { /* Track include filename */
	msg_tail("(included at line");
	msg_tail(ulongtostr((unsigned long)top_linenum));
	msg_tail("in");
	msg_tail(topfile);
	msg_tail(")");
    }
}


void
#if HAVE_STDC
sort_gsymbols (Gsymtab **glist, int n)   /* same as sort_lsymbols */
#else /* K&R style */
sort_gsymbols ( glist,n )
	Gsymtab *glist[];
	int n;
#endif /* HAVE_STDC */
{
	int i,j,swaps;

	for (i=0; i<n; i++ ){
	    swaps = 0;
	    for  (j=n-1; j>=i+1; j--){
		if ((strcmp (glist[j-1]->name, glist[j]->name)) >0) {
		    Gsymtab *temp = glist[j-1]; /* swap ptrs j and j-1 */
		    glist[j-1] = glist[j];
		    glist[j] = temp;
		    swaps++;
		}
	    }
	    if (swaps == 0) break;
	}


}