parse.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 871 行 · 第 1/2 页
C
871 行
/****************************************************************************
*
* Open Watcom Project
*
* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
* ========================================================================
*
* This file contains Original Code and/or Modifications of Original
* Code as defined in and that are subject to the Sybase Open Watcom
* Public License version 1.0 (the 'License'). You may not use this file
* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
* provided with the Original Code and Modifications, and is also
* available at www.sybase.com/developer/opensource.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
* NON-INFRINGEMENT. Please see the License for the specific language
* governing rights and limitations under the License.
*
* ========================================================================
*
* Description: Command line parsing for LINK clone tool.
*
****************************************************************************/
#include <stdio.h>
#include <ctype.h>
#include <stdlib.h>
#include <string.h>
#include "link.h"
#include "cmdline.h"
#include "cmdscan.h"
#include "context.h"
#include "error.h"
#include "file.h"
#include "memory.h"
#include "message.h"
#include "parse.h"
/*
* Initialize the OPT_STORAGE structure.
*/
void InitParse( OPT_STORAGE *cmdOpts )
/************************************/
{
OPT_INIT( cmdOpts );
}
/*
* Destroy the OPT_STORAGE structure.
*/
void FiniParse( OPT_STORAGE *cmdOpts )
/************************************/
{
OPT_FINI( cmdOpts );
}
/*
* Gripe about a command line error.
*/
static void cmd_line_error( void )
/********************************/
{
char * str;
GoToMarkContext();
str = CmdScanString();
Warning( "Ignoring invalid option '%s'", str );
}
/*
* Parse the command string contained in the current context.
*/
void CmdStringParse( OPT_STORAGE *cmdOpts, int *itemsParsed )
/***********************************************************/
{
int ch;
char * filename;
for( ;; ) {
/*** Find the start of the next item ***/
CmdScanWhitespace();
ch = GetCharContext();
if( ch == '\0' ) break;
MarkPosContext(); /* mark start of switch */
/*** Handle switches, command files, and input files ***/
if( ch == '-' || ch == '/' ) { /* switch */
if( OPT_PROCESS( cmdOpts ) != 0 ) {
cmd_line_error();
}
} else if( ch == '@' ) { /* command file */
filename = CmdScanFileNameWithoutQuotes();
PushContext();
if( OpenFileContext( filename ) ) {
FatalError( "Cannot open '%s'.", filename );
}
FreeMem( filename );
CmdStringParse( cmdOpts, itemsParsed );
PopContext();
} else { /* input file */
UngetCharContext();
filename = CmdScanFileName();
AddFile( TYPE_DEFAULT_FILE, filename );
FreeMem( filename );
}
(*itemsParsed)++;
}
CloseContext();
}
/*
* Ensure argument to /ALIGN was valid.
*/
static void check_align( unsigned *p )
/************************************/
{
int numFound = 0;
unsigned temp = *p;
/*** Count the number of bits that are on ***/
while( temp != 0 ) {
if( temp & 0x0001 ) numFound++;
temp >>= 1;
}
/*** Gripe if the number isn't a power of two ***/
if( numFound != 1 ) {
Warning( "Invalid argument %u to /ALIGN -- assuming 4096", *p );
*p = 4096;
}
}
#define skip_white(s) while (*s && isspace(*s)) ++s;
#define skip_nonwhite(s) while (*s && !isspace(*s)) ++s;
/*
* Add another string to an OPT_STRING.
*/
static void add_string( OPT_STRING **p, char *str, char quote )
/*************************************************************/
{
OPT_STRING * buf;
OPT_STRING * curElem;
int len;
int add_quote = 0;
len = strlen(str);
if( quote != 0 ) {
do {
if( str[0] == '"' && str[len-1] == '"' ) break;
if( str[0] == '\'' && str[len-1] == '\'' ) break;
len += 2;
add_quote = 1;
} while( 0 );
}
/*** Make a new list item ***/
buf = AllocMem( sizeof(OPT_STRING) + len );
if( add_quote ) {
buf->data[0] = quote;
strcpy( &(buf->data[1]), str );
buf->data[len-1] = quote;
buf->data[len] = '\0';
} else {
strcpy( buf->data, str );
}
buf->next = NULL;
/*** Put it at the end of the list ***/
if( *p == NULL ) {
*p = buf;
} else {
curElem = *p;
while( curElem->next != NULL ) curElem = curElem->next;
curElem->next = buf;
}
}
/*
* Destroy an OPT_STRING.
*/
static void OPT_CLEAN_STRING( OPT_STRING **p )
/********************************************/
{
OPT_STRING * s;
while( *p != NULL ) {
s = *p;
*p = s->next;
FreeMem( s );
}
}
/*
* Parse the /BASE:@{filename} option.
*/
static int parse_base_from_file(OPT_STRING **p, char *str)
{
FILE *stream;
char *buffer;
char *keyword;
char *s;
char *b;
int keylen;
buffer = AllocMem(255);
for (s = str + 1, b = buffer; *s && (*s != ',') && !isspace(*s); )
*b++ = *s++;
if (*s++ != ',')
{
FreeMem(buffer);
FatalError("/BASE:@{filename} requires valid filename");
return 0;
}
*b = 0x00;
keyword = s;
skip_nonwhite(s);
keylen = s - keyword;
stream = fopen(buffer, "r");
if (!stream)
{
FreeMem(buffer);
FatalError("/BASE:@{filename} requires a valid filename");
return 0;
}
while (fgets(buffer, 255, stream))
{
s = buffer;
skip_white(s);
if (*s == ';' || *s == '\0' )
continue;
if (!strnicmp(s, keyword, keylen) && isspace(s[keylen]))
{
s += keylen + 1;
skip_white(s);
if (!*s)
{
FreeMem(buffer);
fclose(stream);
FatalError("/BASE file contains invalid offset");
return 0;
}
b = s;
skip_nonwhite(b);
*b = 0x00;
OPT_CLEAN_STRING(p);
add_string(p, s, 0);
FreeMem(buffer);
fclose(stream);
return 1;
}
}
FreeMem(buffer);
fclose(stream);
FatalError("/BASE file does not contain keyword");
return 0;
} /* parse_base_from_file() */
/*
* Parse the /BASE option.
*/
static int parse_base( OPT_STRING **p )
/*************************************/
{
char *str;
if (!CmdScanRecogChar(':'))
{
FatalError("/BASE requires an argument");
return 0;
}
str = CmdScanString();
if (str == NULL)
{
FatalError("/BASE requires an argument");
return 0;
}
if (*str == '@')
{
return parse_base_from_file(p, str);
}
else
{
OPT_CLEAN_STRING(p);
add_string(p, str, 0);
}
return 1;
} /* parse_base() */
/*
* For the /optName option, read in :string and store the string into the
* given OPT_STRING. If onlyOne is non-zero, any previous string in p will
* be deleted. If quote is non-zero, make sure the string is quoted.
* Use quote if there aren't any quotes already.
*/
static int do_string_parse( OPT_STRING **p, char *optName, int onlyOne,
/*********************************************************************/
int quote )
{
char * str;
if( !CmdScanRecogChar( ':' ) ) {
FatalError( "/%s requires an argument", optName );
return( 0 );
}
str = CmdScanString();
if( str == NULL ) {
FatalError( "/%s requires an argument", optName );
return( 0 );
}
if( onlyOne ) OPT_CLEAN_STRING( p );
add_string( p, str, quote );
return( 1 );
}
/*
* Parse the /COMMENT option.
*/
static int parse_comment( OPT_STRING **p )
/****************************************/
{
return( do_string_parse( p, "COMMENT", 0, 0 ) );
}
/*
* Parse the /DEBUG option.
*/
static int parse_debug( OPT_STRING **p )
/**************************************/
{
char * str;
int ch;
ch = GetCharContext();
if( ch != ':' ) { /* optional :<type> */
if( ch != '\0' ) UngetCharContext();
} else {
str = CmdScanString();
if( str == NULL ) {
FatalError( "Missing argument for /DEBUG" );
return( 0 );
} else {
add_string( p, str, 0 );
FreeMem( str );
}
}
return( 1 );
}
/*
* Parse the /DEBUGTYPE option.
*/
static int parse_debugtype( OPT_STRING **p )
/******************************************/
{
return( do_string_parse( p, "DEBUGTYPE", 0, 0 ) );
}
/*
* Parse the /DEF option.
*/
static int parse_def( OPT_STRING **p )
/************************************/
{
return( do_string_parse( p, "DEF", 1, 0 ) );
}
/*
* Parse the /DEFAULTLIB option.
*/
static int parse_defaultlib( OPT_STRING **p )
/*******************************************/
{
return( do_string_parse( p, "DEFAULTLIB", 0, 0 ) );
}
/*
* Parse the /ENTRY option.
*/
static int parse_entry( OPT_STRING **p )
/**************************************/
{
return( do_string_parse( p, "ENTRY", 1, '\'' ) );
}
/*
* Parse the /EXETYPE option.
*/
static int parse_exetype( OPT_STRING **p )
/****************************************/
{
return( do_string_parse( p, "EXETYPE", 1, 0 ) );
}
/*
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?