* harbour/hbgtmk.sh
* harbour/make_rpm.sh
* harbour/harbour.spec
* added flex to dependences
* added --without gpl swich to exclude code which needs 3-rd party
GPL libs (GPM, SLANG) - necessary for people who want to create
commercial closed source application
* harbour/make_gnu.sh
* modified DJGPP detection - try to use env var first
* set HB_GT_LIB depending on platform information
* check PREFIX env var for default installation
* harbour/make_tgz.sh
* modified to work with DJGPP and 8.3 file names
* harbour/bin/hb-func.sh
* DJGPP support
* harbour/bin/pack_src.sh
* autodetect Harbour version number
* harbour/config/linux/gcc.cf
* added small comment
+ harbour/config/linux/owatcom.cf
+ OpenWatcom support on Linux
* harbour/config/w32/global.cf
* harbour/config/w32/mingw32.cf
* small modification in clean procedure
+ added Phil's modification for MINGW cross compilation in Linux and BSD
* harbour/source/compiler/gencobj.c
+ added Phil's modification for MINGW cross compilation in Linux and BSD
* harbour/include/hbsetup.h
* added OS_HAS_DRIVE_LETTER, OS_PATH_DELIMITER_STRING, OS_FILE_MASK,
OS_DRIVE_DELIMITER and HOST_OS_UNIX_COMPATIBLE
* modified for OW on Linux and MINGW cross compilation
* added snprintf macro for compilers which do not support it
* harbour/source/rtl/mod.c
! fixed bug in mod(a,b) when a/b exceeds LONG limit
* harbour/source/rtl/filesys.c
* removed HB_FS_DRIVE_LETTER - use OS_HAS_DRIVE_LETTER
* modified for OW on Linux and MINGW cross compilation
* harbour/contrib/libct/files.c
* harbour/include/hb_io.h
* harbour/include/hbdefs.h
* harbour/source/rtl/diskspac.c
* harbour/source/rtl/disksphb.c
* harbour/source/rtl/fssize.c
* harbour/source/rtl/fstemp.c
* harbour/source/rtl/hbffind.c
* harbour/source/rtl/net.c
* harbour/source/rtl/seconds.c
* harbour/source/rtl/gtcrs/Makefile
* harbour/source/rtl/gtcrs/kbdcrs.c
* harbour/source/rtl/gtpca/gtpca.c
* harbour/source/rtl/gtsln/Makefile
* harbour/source/rtl/gtsln/gtsln.c
* harbour/source/rtl/gtsln/kbsln.c
* harbour/source/rtl/gtstd/gtstd.c
* harbour/source/vm/hvm.c
* harbour/source/vm/mainstd.c
* harbour/source/vm/mainwin.c
* modified for OW on Linux and MINGW cross compilation
* harbour/source/vm/memvars.c
- removed unnecessary checking for s_globalTable == NULL in
hb_memvarValueDecRef() - in Harbour it cannot happen
* harbour/source/vm/itemapi.c
- removed snprintf macros (see hbsetup.h)
* fixed possible GPF in hb_itemCopyC()
* modified hb_itemGetPtr() to check item type
279 lines
7.6 KiB
C
279 lines
7.6 KiB
C
/*
|
||
* $Id$
|
||
*/
|
||
|
||
/*
|
||
* Harbour Project source code:
|
||
* Native compiler object module generation from Harbour C output.
|
||
*
|
||
* Copyright 2001 Jos‚ Lal¡n <dezac@corevia.com>
|
||
* www - http://www.harbour-project.org
|
||
*
|
||
* 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.
|
||
*
|
||
* You should have received a copy of the GNU General Public License
|
||
* along with this program; if not, write to the Free Software
|
||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA (or visit
|
||
* their web site at http://www.gnu.org/).
|
||
*
|
||
*/
|
||
|
||
#include "hbcomp.h"
|
||
#include "hb_io.h"
|
||
|
||
/* Prototype */
|
||
static char * hb_searchpath( const char *, char *, char * );
|
||
|
||
#define HB_CFG_FILENAME "harbour.cfg"
|
||
|
||
/* QUESTION: Allocate buffer dynamically ? */
|
||
#define HB_CFG_LINE_LEN 100
|
||
|
||
/*--------------------------------------------------------------------------*/
|
||
|
||
/* Builds platform dependant object module from Harbour C output */
|
||
void hb_compGenCObj( PHB_FNAME pFileName )
|
||
{
|
||
char szFileName[ _POSIX_PATH_MAX ];
|
||
char szLine[ HB_CFG_LINE_LEN ];
|
||
char szCompiler[ HB_CFG_LINE_LEN ] = "";
|
||
char szOptions[ HB_CFG_LINE_LEN ] = "";
|
||
char szCommandLine[ HB_CFG_LINE_LEN * 2 ];
|
||
char szOutPath[ _POSIX_PATH_MAX ] = "\0";
|
||
#if defined( HOST_OS_UNIX_COMPATIBLE )
|
||
char szDefaultUnixPath[ _POSIX_PATH_MAX ] = "/etc:/usr/local/etc";
|
||
#define HB_NULL_STR " > /dev/null"
|
||
#define HB_ACCESS_FLAG F_OK
|
||
#elif defined( OS_DOS_COMPATIBLE )
|
||
#define HB_NULL_STR " >nul"
|
||
#define HB_ACCESS_FLAG 0
|
||
#endif
|
||
FILE * yyc;
|
||
char * pszCfg;
|
||
char * pszEnv;
|
||
BOOL bVerbose = FALSE; /* Don't show C compiler messages (default). */
|
||
BOOL bDelTmp = TRUE; /* Delete intermediate C file (default). */
|
||
int iSuccess;
|
||
|
||
/* First pass: build the C output */
|
||
|
||
/* Force file extension to avoid collisions when called from a make utility */
|
||
pFileName->szExtension = ".c";
|
||
hb_fsFNameMerge( szFileName, pFileName );
|
||
hb_compGenCCode( hb_comp_pFileName );
|
||
|
||
/* Begin second pass */
|
||
|
||
/* Set up things */
|
||
#if defined( OS_DOS_COMPATIBLE )
|
||
pszEnv = hb_getenv( "PATH" );
|
||
#elif defined( OS_UNIX_COMPATIBLE )
|
||
pszEnv = szDefaultUnixPath;
|
||
#else
|
||
pszEnv = NULL;
|
||
#endif
|
||
|
||
/* Grab space */
|
||
pszCfg = ( char * ) hb_xgrab( /*strlen( pszEnv )*/ _POSIX_PATH_MAX );
|
||
|
||
if( pszEnv && pszEnv[ 0 ] != '\0' && *hb_searchpath( HB_CFG_FILENAME, pszEnv, pszCfg ) )
|
||
{
|
||
|
||
yyc = fopen( pszCfg, "rt" );
|
||
if( ! yyc )
|
||
{
|
||
#if 0
|
||
/* QUESTION: Add a new error to Harbour ?
|
||
hb_compGenError( hb_comp_szErrors, 'E', HB_COMP_ERR_OPEN_CFG, szFileName, NULL );
|
||
*/
|
||
#else
|
||
printf( "\nError: Can't find %s file.\n", HB_CFG_FILENAME );
|
||
return;
|
||
#endif
|
||
}
|
||
|
||
while( fgets( szLine, HB_CFG_LINE_LEN, yyc ) != NULL )
|
||
{
|
||
ULONG ulLen;
|
||
char * szStr = szLine;
|
||
char * szToken;
|
||
|
||
/* Trim left */
|
||
while( HB_ISSPACE( *szStr ) )
|
||
szStr++;
|
||
|
||
/* Trim right */
|
||
ulLen = strlen( szStr );
|
||
while( ulLen && HB_ISSPACE( szStr[ ulLen - 1 ] ) )
|
||
ulLen--;
|
||
|
||
szStr[ ulLen ] = '\0';
|
||
/* TODO: Check for comments within macros, i.e: CC=bcc32 #comment */
|
||
|
||
if( *szStr )
|
||
{
|
||
szToken = strchr( szStr, '=' );
|
||
|
||
if( szToken )
|
||
{
|
||
*szToken++ = '\0';
|
||
if ( *szToken )
|
||
{
|
||
/* Checks compiler name */
|
||
if( ! hb_stricmp( szStr, "CC" ) )
|
||
{
|
||
sprintf( szCompiler, "%s", szToken );
|
||
}
|
||
/* Checks optional switches */
|
||
else if( ! hb_stricmp( szStr, "CFLAGS" ) )
|
||
{
|
||
sprintf( szOptions, "%s", szToken );
|
||
}
|
||
/* Wanna see C compiler output ? */
|
||
else if( ! hb_stricmp( szStr, "VERBOSE" ) )
|
||
{
|
||
if( ! hb_stricmp( szToken, "YES" ) )
|
||
bVerbose = TRUE;
|
||
}
|
||
/* Delete intermediate C file ? */
|
||
else if( ! hb_stricmp( szStr, "DELTMP" ) )
|
||
{
|
||
if( ! hb_stricmp( szToken, "NO" ) )
|
||
bDelTmp = FALSE;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
fclose( yyc );
|
||
}
|
||
|
||
#if defined( OS_DOS_COMPATIBLE )
|
||
{
|
||
if( pszEnv )
|
||
hb_xfree( ( void * ) pszEnv );
|
||
}
|
||
#endif
|
||
|
||
|
||
if( ! hb_comp_bQuiet )
|
||
{
|
||
printf( "Building object module output for \'%s\'...", szFileName );
|
||
fflush( stdout );
|
||
}
|
||
|
||
/* Check if -o<path> was used */
|
||
if( hb_comp_pOutPath )
|
||
{
|
||
PHB_FNAME pOut = hb_fsFNameSplit( ( char * ) szFileName );
|
||
char pszTemp[ _POSIX_PATH_MAX ] = "";
|
||
|
||
if( hb_comp_pOutPath->szPath )
|
||
pOut->szPath = hb_comp_pOutPath->szPath;
|
||
|
||
#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__WATCOMC__)
|
||
pOut->szExtension = ".obj";
|
||
#else
|
||
pOut->szExtension = ".o"; /* Don't know if we can hardcode it for Un*x */
|
||
#endif
|
||
hb_fsFNameMerge( pszTemp, pOut );
|
||
|
||
#if defined(_MSC_VER)
|
||
strcat( szOutPath, "-Fo" );
|
||
#elif defined(__WATCOMC__)
|
||
strcat( szOutPath, "-fo=" );
|
||
#else
|
||
strcat( szOutPath, "-o" );
|
||
#endif
|
||
|
||
strcat( szOutPath, pszTemp );
|
||
|
||
hb_xfree( pOut );
|
||
}
|
||
|
||
if( *szCompiler )
|
||
{
|
||
sprintf( szCommandLine, "%s %s %s %s", szCompiler, szOptions, szOutPath, szFileName );
|
||
|
||
if( bVerbose )
|
||
{
|
||
printf( "\n%s\n", szCommandLine ) ;
|
||
}
|
||
else
|
||
{
|
||
strcat( szCommandLine, HB_NULL_STR );
|
||
}
|
||
|
||
/* Compile it! */
|
||
iSuccess = ( system( szCommandLine ) != -1 );
|
||
|
||
/* Delete intermediate .c file */
|
||
/* QUESTION: Leave this file if C compiler fails ? */
|
||
if( bDelTmp ) /* && iSuccess ) */
|
||
unlink( ( char * ) szFileName );
|
||
|
||
if( ! hb_comp_bQuiet )
|
||
{
|
||
if( iSuccess )
|
||
printf( "Done.\n" );
|
||
else
|
||
printf( "\nFailed to execute: \"%s\"\n", szCommandLine );
|
||
}
|
||
}
|
||
else
|
||
{
|
||
printf( "\nError: No compiler defined in %s\n", HB_CFG_FILENAME );
|
||
}
|
||
|
||
if( pszCfg )
|
||
hb_xfree( pszCfg );
|
||
}
|
||
|
||
static char * hb_searchpath( const char * pszFile, char * pszEnv, char * pszCfg )
|
||
{
|
||
char * pszPath;
|
||
char pszDelim[2] = { OS_PATH_LIST_SEPARATOR, '\0'};
|
||
BOOL bFound = FALSE;
|
||
|
||
/* Check current dir first */
|
||
if( access( ( const char * ) pszFile, HB_ACCESS_FLAG ) == 0 )
|
||
{
|
||
sprintf( pszCfg, "%s", pszFile );
|
||
return ( char * ) pszFile;
|
||
}
|
||
else
|
||
{
|
||
/* Check if pszFile exists somewhere in the path */
|
||
pszPath = strtok( pszEnv, pszDelim );
|
||
if( pszPath )
|
||
{
|
||
while( pszPath )
|
||
{
|
||
sprintf( pszCfg, "%s%c%s", pszPath, OS_PATH_DELIMITER, pszFile );
|
||
if( access( ( const char * ) pszCfg, HB_ACCESS_FLAG ) == 0 )
|
||
{
|
||
bFound = TRUE;
|
||
break;
|
||
}
|
||
else
|
||
pszPath = strtok( NULL, pszDelim );
|
||
}
|
||
}
|
||
}
|
||
|
||
/* If not found, make sure to return a NULL string */
|
||
if( ! bFound )
|
||
sprintf( pszCfg, "%s", "" );
|
||
|
||
return ( char * ) pszCfg;
|
||
}
|