// Compiler implementation of the D programming language // Copyright (c) 1999-2012 by Digital Mars // All Rights Reserved // written by Walter Bright // http://www.digitalmars.com // License for redistribution is by either the Artistic License // in artistic.txt, or the GNU General Public License in gnu.txt. // See the included readme.txt for details. #ifndef DMD_MARS_H #define DMD_MARS_H #ifdef __DMC__ #pragma once #endif /* It is very important to use version control macros correctly - the idea is that host and target are independent. If these are done correctly, cross compilers can be built. The host compiler and host operating system are also different, and are predefined by the host compiler. The ones used in dmd are: Macros defined by the compiler, not the code: Compiler: __DMC__ Digital Mars compiler _MSC_VER Microsoft compiler __GNUC__ Gnu compiler __clang__ Clang compiler Host operating system: _WIN32 Microsoft NT, Windows 95, Windows 98, Win32s, Windows 2000, Win XP, Vista _WIN64 Windows for AMD64 linux Linux __APPLE__ Mac OSX __FreeBSD__ FreeBSD __OpenBSD__ OpenBSD __sun Solaris, OpenSolaris, SunOS, OpenIndiana, etc For the target systems, there are the target operating system and the target object file format: Target operating system: TARGET_WINDOS Covers 32 bit windows and 64 bit windows TARGET_LINUX Covers 32 and 64 bit linux TARGET_OSX Covers 32 and 64 bit Mac OSX TARGET_FREEBSD Covers 32 and 64 bit FreeBSD TARGET_OPENBSD Covers 32 and 64 bit OpenBSD TARGET_SOLARIS Covers 32 and 64 bit Solaris TARGET_NET Covers .Net It is expected that the compiler for each platform will be able to generate 32 and 64 bit code from the same compiler binary. Target object module format: OMFOBJ Intel Object Module Format, used on Windows ELFOBJ Elf Object Module Format, used on linux, FreeBSD, OpenBSD and Solaris MACHOBJ Mach-O Object Module Format, used on Mac OSX There are currently no macros for byte endianness order. */ #include #include #include #include #include #if IN_LLVM #include "llvm/ADT/Triple.h" #endif #ifdef __DMC__ #ifdef DEBUG #undef assert #define assert(e) (static_cast((e) || (printf("assert %s(%d) %s\n", __FILE__, __LINE__, #e), halt()))) #endif #endif #ifdef DEBUG #define UNITTEST 1 #endif void unittests(); #ifndef IS_PRINTF # ifdef __GNUC__ # define IS_PRINTF(FMTARG) __attribute((__format__ (__printf__, (FMTARG), (FMTARG)+1) )) # else # define IS_PRINTF(FMTARG) # endif #endif #ifdef IN_GCC /* Changes for the GDC compiler by David Friedman */ #endif #define DMDV1 0 #define DMDV2 1 // Version 2.0 features #define STRUCTTHISREF DMDV2 // if 'this' for struct is a reference, not a pointer #define SNAN_DEFAULT_INIT DMDV2 // if floats are default initialized to signalling NaN #define SARRAYVALUE DMDV2 // static arrays are value types #define MODULEINFO_IS_STRUCT DMDV2 // if ModuleInfo is a struct rather than a class #define BUG6652 1 // Making foreach range statement parameter non-ref in default // 1: Modifying iteratee in body is warned with -w switch // 2: Modifying iteratee in body is error without -d switch // Set if C++ mangling is done by the front end #define CPP_MANGLE (DMDV2 && (TARGET_LINUX || TARGET_OSX || TARGET_FREEBSD || TARGET_OPENBSD || TARGET_SOLARIS || IN_LLVM)) /* Other targets are TARGET_LINUX, TARGET_OSX, TARGET_FREEBSD, TARGET_OPENBSD and * TARGET_SOLARIS, which are * set on the command line via the compiler makefile. */ #if _WIN32 #ifndef TARGET_WINDOS #define TARGET_WINDOS 1 // Windows dmd generates Windows targets #endif #ifndef OMFOBJ #define OMFOBJ TARGET_WINDOS #endif #endif #if TARGET_LINUX || TARGET_FREEBSD || TARGET_OPENBSD || TARGET_SOLARIS #ifndef ELFOBJ #define ELFOBJ 1 #endif #endif #if TARGET_OSX #ifndef MACHOBJ #define MACHOBJ 1 #endif #endif struct OutBuffer; // Can't include arraytypes.h here, need to declare these directly. template struct ArrayBase; typedef ArrayBase Identifiers; typedef ArrayBase Strings; #if IN_LLVM enum OUTPUTFLAG { OUTPUTFLAGno, OUTPUTFLAGdefault, // for the .o default OUTPUTFLAGset // for -output }; typedef unsigned char ubyte; #endif // Put command line switches in here struct Param { bool obj; // write object file bool link; // perform link #if !IN_LLVM char dll; // generate shared dynamic library char lib; // write library file instead of object file(s) char multiobj; // break one object file into multiple ones char oneobj; // write one object file instead of multiple ones char trace; // insert profiling hooks char quiet; // suppress non-error messages #endif bool verbose; // verbose compile bool vtls; // identify thread local variables ubyte symdebug; // insert debug symbolic information #if !IN_LLVM char alwaysframe; // always emit standard stack frame char optimize; // run optimizer #endif char map; // generate linker .map file bool is64bit; // generate 64 bit code #if !IN_LLVM char isLinux; // generate code for linux char isOSX; // generate code for Mac OSX char isWindows; // generate code for Windows char isFreeBSD; // generate code for FreeBSD char isOPenBSD; // generate code for OpenBSD char isSolaris; // generate code for Solaris char scheduler; // which scheduler to use #endif ubyte useDeprecated; // 0: don't allow use of deprecated features // 1: silently allow use of deprecated features // 2: warn about the use of deprecated features bool useAssert; // generate runtime code for assert()'s bool useInvariants; // generate class invariant checks bool useIn; // generate precondition checks bool useOut; // generate postcondition checks #if IN_LLVM bool useArrayBounds; #else char useArrayBounds; // 0: no array bounds checks // 1: array bounds checks for safe functions only // 2: array bounds checks for all functions #endif bool noboundscheck; // no array bounds checking at all bool useSwitchError; // check for switches without a default bool useUnitTests; // generate unittest code bool useInline; // inline expand functions #if !IN_LLVM char release; // build release version char preservePaths; // !=0 means don't strip path from source file #endif ubyte warnings; // 0: enable warnings // 1: warnings as errors // 2: informational warnings (no errors) #if !IN_LLVM char pic; // generate position-independent-code for shared libs char cov; // generate code coverage data char nofloat; // code should not pull in floating point support #endif ubyte Dversion; // D version number bool ignoreUnsupportedPragmas; // rather than error on them bool enforcePropertySyntax; #if !IN_LLVM char betterC; // be a "better C" compiler; no dependency on D runtime #endif char *argv0; // program name Strings *imppath; // array of char*'s of where to look for import modules Strings *fileImppath; // array of char*'s of where to look for file import modules char *objdir; // .obj/.lib file output directory char *objname; // .obj file output name bool doDocComments; // process embedded documentation comments char *docdir; // write documentation file to docdir directory char *docname; // write documentation file to docname Strings *ddocfiles; // macro include files for Ddoc bool doHdrGeneration; // process embedded documentation comments char *hdrdir; // write 'header' file to docdir directory char *hdrname; // write 'header' file to docname bool doXGeneration; // write JSON file char *xfilename; // write JSON file to xfilename unsigned debuglevel; // debug level Strings *debugids; // debug identifiers unsigned versionlevel; // version level Strings *versionids; // version identifiers bool dump_source; Strings *defaultlibnames; // default libraries for non-debug builds Strings *debuglibnames; // default libraries for debug builds char *moduleDepsFile; // filename for deps output OutBuffer *moduleDeps; // contents to be written to deps file #if IN_DMD // Hidden debug switches bool debuga; bool debugb; bool debugc; bool debugf; bool debugr; bool debugw; bool debugx; bool debugy; #endif bool run; // run resulting executable #if !IN_LLVM size_t runargs_length; char** runargs; // arguments for executable #endif // Linker stuff Strings *objfiles; Strings *linkswitches; Strings *libfiles; char *deffile; char *resfile; char *exefile; char *mapfile; #if IN_LLVM // LDC stuff OUTPUTFLAG output_ll; OUTPUTFLAG output_bc; OUTPUTFLAG output_s; OUTPUTFLAG output_o; bool useInlineAsm; bool verbose_cg; bool useAvailableExternally; // target stuff llvm::Triple targetTriple; // Codegen cl options bool singleObj; bool disableRedZone; #endif }; typedef unsigned structalign_t; #define STRUCTALIGN_DEFAULT ~0 // magic value means "match whatever the underlying C compiler does" // other values are all powers of 2 struct Global { const char *mars_ext; const char *sym_ext; const char *obj_ext; #if IN_LLVM const char *obj_ext_alt; const char *ll_ext; const char *bc_ext; const char *s_ext; #endif const char *lib_ext; const char *dll_ext; const char *doc_ext; // for Ddoc generated files const char *ddoc_ext; // for Ddoc macro include files const char *hdr_ext; // for D 'header' import files const char *json_ext; // for JSON files const char *map_ext; // for .map files const char *copyright; const char *written; Strings *path; // Array of char*'s which form the import lookup path Strings *filePath; // Array of char*'s which form the file import lookup path structalign_t structalign; // default alignment for struct fields const char *version; #if IN_LLVM char *ldc_version; char *llvm_version; #endif Param params; unsigned errors; // number of errors reported so far unsigned warnings; // number of warnings reported so far unsigned gag; // !=0 means gag reporting of errors & warnings unsigned gaggedErrors; // number of errors reported while gagged /* Gagging can either be speculative (is(typeof()), etc) * or because of forward references */ unsigned speculativeGag; // == gag means gagging is for is(typeof); bool isSpeculativeGagging(); // Start gagging. Return the current number of gagged errors unsigned startGagging(); /* End gagging, restoring the old gagged state. * Return true if errors occured while gagged. */ bool endGagging(unsigned oldGagged); Global(); }; extern Global global; /* Set if Windows Structured Exception Handling C extensions are supported. * Apparently, VC has dropped support for these? */ #define WINDOWS_SEH (_WIN32 && !defined(__MINGW32__)) #include "longdouble.h" #ifdef __DMC__ #include typedef _Complex long double complex_t; #else #ifndef IN_GCC #include "complex_t.h" #endif #ifdef __APPLE__ //#include "complex.h"//This causes problems with include the c++ and not the C "complex.h" #endif #endif // Be careful not to care about sign when using dinteger_t //typedef uint64_t integer_t; typedef uint64_t dinteger_t; // use this instead of integer_t to // avoid conflicts with system #include's // Signed and unsigned variants typedef int64_t sinteger_t; typedef uint64_t uinteger_t; typedef int8_t d_int8; typedef uint8_t d_uns8; typedef int16_t d_int16; typedef uint16_t d_uns16; typedef int32_t d_int32; typedef uint32_t d_uns32; typedef int64_t d_int64; typedef uint64_t d_uns64; typedef float d_float32; typedef double d_float64; typedef longdouble d_float80; typedef d_uns8 d_char; typedef d_uns16 d_wchar; typedef d_uns32 d_dchar; #ifdef IN_GCC #include "d-gcc-real.h" #else typedef longdouble real_t; #endif // Modify OutBuffer::writewchar to write the correct size of wchar #if _WIN32 #define writewchar writeword #else // This needs a configuration test... #define writewchar write4 #endif #ifdef IN_GCC #include "d-gcc-complex_t.h" #endif struct Module; //typedef unsigned Loc; // file location struct Loc { const char *filename; unsigned linnum; Loc() { linnum = 0; filename = NULL; } Loc(int x) { linnum = x; filename = NULL; } Loc(Module *mod, unsigned linnum); char *toChars(); bool equals(const Loc& loc); }; #ifndef GCC_SAFE_DMD #define TRUE 1 #define FALSE 0 #endif #define INTERFACE_OFFSET 0 // if 1, put classinfo as first entry // in interface vtbl[]'s #define INTERFACE_VIRTUAL 0 // 1 means if an interface appears // in the inheritance graph multiple // times, only one is used enum LINK { LINKdefault, LINKd, LINKc, LINKcpp, LINKwindows, LINKpascal, #if IN_LLVM LINKintrinsic, #endif }; enum DYNCAST { DYNCAST_OBJECT, DYNCAST_EXPRESSION, DYNCAST_DSYMBOL, DYNCAST_TYPE, DYNCAST_IDENTIFIER, DYNCAST_TUPLE, DYNCAST_PARAMETER, }; enum MATCH { MATCHnomatch, // no match MATCHconvert, // match with conversions #if DMDV2 MATCHconst, // match with conversion to const #endif MATCHexact // exact match }; typedef uint64_t StorageClass; void warning(Loc loc, const char *format, ...); void deprecation(Loc loc, const char *format, ...); void error(Loc loc, const char *format, ...); void errorSupplemental(Loc loc, const char *format, ...); void verror(Loc loc, const char *format, va_list ap, const char *p1 = NULL, const char *p2 = NULL, const char *header = "Error: "); void vwarning(Loc loc, const char *format, va_list); void verrorSupplemental(Loc loc, const char *format, va_list ap); void verrorPrint(Loc loc, const char *header, const char *format, va_list ap, const char *p1 = NULL, const char *p2 = NULL); void vdeprecation(Loc loc, const char *format, va_list ap, const char *p1 = NULL, const char *p2 = NULL); #if defined(__GNUC__) || defined(__clang__) __attribute__((noreturn)) #endif void fatal(); void err_nomem(); #if IN_LLVM void error(const char *format, ...) IS_PRINTF(1); void warning(const char *format, ...) IS_PRINTF(1); #else int runLINK(); void deleteExeFile(); int runProgram(); const char *inifile(const char *argv0, const char *inifile, const char* envsectionname); #endif void halt(); #if !IN_LLVM void util_progress(); #endif /*** Where to send error messages ***/ #if defined(IN_GCC) || IN_LLVM #define stdmsg stderr #else #define stdmsg stderr #endif #if !IN_LLVM struct Dsymbol; class Library; struct File; void obj_start(char *srcfile); void obj_end(Library *library, File *objfile); void obj_append(Dsymbol *s); void obj_write_deferred(Library *library); #endif const char *importHint(const char *s); #endif /* DMD_MARS_H */