2011-05-16 13:12:07 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <string.h>
|
2019-08-17 18:32:15 +00:00
|
|
|
#include <stdarg.h>
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
2018-09-05 19:49:02 +00:00
|
|
|
#define strcasecmp(_String1, _String2) _stricmp(_String1, _String2)
|
|
|
|
#define strncasecmp(_String1, _String2, _MaxCount) _strnicmp(_String1, _String2, _MaxCount)
|
2011-05-16 13:12:07 +00:00
|
|
|
#endif
|
|
|
|
|
2015-02-04 23:56:23 +00:00
|
|
|
#define ARRAYSIZE(a) (sizeof(a) / sizeof((a)[0]))
|
|
|
|
|
2013-11-17 22:11:04 +00:00
|
|
|
typedef struct _STRING
|
|
|
|
{
|
|
|
|
const char *buf;
|
|
|
|
int len;
|
|
|
|
} STRING, *PSTRING;
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
STRING strName;
|
|
|
|
STRING strTarget;
|
2011-05-16 13:12:07 +00:00
|
|
|
int nCallingConvention;
|
|
|
|
int nOrdinal;
|
|
|
|
int nStackBytes;
|
|
|
|
int nArgCount;
|
|
|
|
int anArgs[30];
|
|
|
|
unsigned int uFlags;
|
2011-07-21 18:39:24 +00:00
|
|
|
int nNumber;
|
2019-08-18 13:32:22 +00:00
|
|
|
unsigned nStartVersion;
|
|
|
|
unsigned nEndVersion;
|
|
|
|
int bVersionIncluded;
|
2011-05-16 13:12:07 +00:00
|
|
|
} EXPORT;
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
#if 0 // Debug helper function
|
|
|
|
void
|
|
|
|
PrintExport(EXPORT *pexp)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "strName='%.*s'\n", pexp->strName.len, pexp->strName.buf);
|
|
|
|
fprintf(stderr, "strName='%.*s'\n", pexp->strTarget.len, pexp->strTarget.buf);
|
|
|
|
fprintf(stderr, "nCallingConvention=%u\n", pexp->nCallingConvention);
|
|
|
|
fprintf(stderr, "nOrdinal=%u\n", pexp->nOrdinal);
|
|
|
|
fprintf(stderr, "nStackBytes=%u\n", pexp->nStackBytes);
|
|
|
|
fprintf(stderr, "nArgCount=%u\n", pexp->nArgCount);
|
|
|
|
fprintf(stderr, "uFlags=0x%x\n", pexp->uFlags);
|
|
|
|
fprintf(stderr, "nNumber=%u\n", pexp->nNumber);
|
|
|
|
fprintf(stderr, "nStartVersion=%u\n", pexp->nStartVersion);
|
|
|
|
fprintf(stderr, "nEndVersion=%u\n", pexp->nEndVersion);
|
|
|
|
fprintf(stderr, "bVersionIncluded=%u\n", pexp->bVersionIncluded);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-06-06 09:58:58 +00:00
|
|
|
enum _ARCH
|
|
|
|
{
|
|
|
|
ARCH_X86,
|
|
|
|
ARCH_AMD64,
|
|
|
|
ARCH_IA64,
|
|
|
|
ARCH_ARM,
|
2021-10-22 15:52:32 +00:00
|
|
|
ARCH_ARM64,
|
2011-06-06 09:58:58 +00:00
|
|
|
ARCH_PPC
|
|
|
|
};
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
typedef int (*PFNOUTLINE)(FILE *, EXPORT *);
|
|
|
|
int gbMSComp = 0;
|
2011-06-26 22:23:08 +00:00
|
|
|
int gbImportLib = 0;
|
2015-02-04 23:56:23 +00:00
|
|
|
int gbNotPrivateNoWarn = 0;
|
2014-09-26 12:43:12 +00:00
|
|
|
int gbTracing = 0;
|
2011-06-06 09:58:58 +00:00
|
|
|
int giArch = ARCH_X86;
|
2011-05-16 13:12:07 +00:00
|
|
|
char *pszArchString = "i386";
|
|
|
|
char *pszArchString2;
|
2015-02-04 23:56:23 +00:00
|
|
|
char *pszSourceFileName = NULL;
|
|
|
|
char *pszDllName = NULL;
|
2011-06-26 22:23:08 +00:00
|
|
|
char *gpszUnderscore = "";
|
2013-11-17 22:11:04 +00:00
|
|
|
int gbDebug;
|
2018-02-18 20:22:52 +00:00
|
|
|
unsigned guOsVersion = 0x502;
|
2013-11-17 22:11:04 +00:00
|
|
|
#define DbgPrint(...) (!gbDebug || fprintf(stderr, __VA_ARGS__))
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
FL_PRIVATE = 1,
|
|
|
|
FL_STUB = 2,
|
|
|
|
FL_NONAME = 4,
|
2014-02-18 20:06:50 +00:00
|
|
|
FL_ORDINAL = 8,
|
2014-09-26 12:43:12 +00:00
|
|
|
FL_NORELAY = 16,
|
|
|
|
FL_RET64 = 32,
|
|
|
|
FL_REGISTER = 64,
|
2011-05-16 13:12:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
CC_STDCALL,
|
|
|
|
CC_CDECL,
|
|
|
|
CC_FASTCALL,
|
2011-10-06 11:10:54 +00:00
|
|
|
CC_THISCALL,
|
2011-05-16 13:12:07 +00:00
|
|
|
CC_EXTERN,
|
|
|
|
CC_STUB,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
ARG_LONG,
|
|
|
|
ARG_PTR,
|
|
|
|
ARG_STR,
|
|
|
|
ARG_WSTR,
|
|
|
|
ARG_DBL,
|
2011-10-06 11:10:54 +00:00
|
|
|
ARG_INT64,
|
2012-09-29 22:58:06 +00:00
|
|
|
ARG_INT128,
|
2011-10-06 11:10:54 +00:00
|
|
|
ARG_FLOAT
|
2011-05-16 13:12:07 +00:00
|
|
|
};
|
|
|
|
|
2015-02-04 23:56:23 +00:00
|
|
|
const char* astrCallingConventions[] =
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
"STDCALL",
|
|
|
|
"CDECL",
|
|
|
|
"FASTCALL",
|
2011-10-06 11:10:54 +00:00
|
|
|
"THISCALL",
|
2011-05-16 13:12:07 +00:00
|
|
|
"EXTERN"
|
|
|
|
};
|
|
|
|
|
2019-06-01 22:11:18 +00:00
|
|
|
/*
|
|
|
|
* List of OLE exports that should be PRIVATE and not be assigned an ordinal.
|
|
|
|
* In case these conditions are not met when linking with MS LINK.EXE, warnings
|
|
|
|
* LNK4104 and LNK4222 respectively are emitted.
|
|
|
|
*/
|
|
|
|
static const char* astrOlePrivateExports[] =
|
2015-02-04 23:56:23 +00:00
|
|
|
{
|
|
|
|
"DllCanUnloadNow",
|
|
|
|
"DllGetClassObject",
|
|
|
|
"DllGetClassFactoryFromClassString",
|
|
|
|
"DllGetDocumentation",
|
|
|
|
"DllInitialize",
|
|
|
|
"DllInstall",
|
|
|
|
"DllRegisterServer",
|
|
|
|
"DllRegisterServerEx",
|
|
|
|
"DllRegisterServerExW",
|
|
|
|
"DllUnload",
|
|
|
|
"DllUnregisterServer",
|
|
|
|
"RasCustomDeleteEntryNotify",
|
|
|
|
"RasCustomDial",
|
|
|
|
"RasCustomDialDlg",
|
|
|
|
"RasCustomEntryDlg",
|
|
|
|
};
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
static
|
|
|
|
int
|
|
|
|
IsSeparator(char chr)
|
|
|
|
{
|
2016-03-28 16:31:32 +00:00
|
|
|
return ((chr <= ',' && chr != '$' && chr != '#') ||
|
2011-05-16 13:12:07 +00:00
|
|
|
(chr >= ':' && chr < '?') );
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
CompareToken(const char *token, const char *comparand)
|
|
|
|
{
|
|
|
|
while (*comparand)
|
|
|
|
{
|
|
|
|
if (*token != *comparand) return 0;
|
|
|
|
token++;
|
|
|
|
comparand++;
|
|
|
|
}
|
2018-02-18 20:22:52 +00:00
|
|
|
if (IsSeparator(comparand[-1])) return 1;
|
2011-05-16 13:12:07 +00:00
|
|
|
if (!IsSeparator(*token)) return 0;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-11-17 22:11:04 +00:00
|
|
|
const char *
|
2011-05-16 13:12:07 +00:00
|
|
|
ScanToken(const char *token, char chr)
|
|
|
|
{
|
|
|
|
while (!IsSeparator(*token))
|
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
if (*token == chr) return token;
|
|
|
|
token++;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *
|
|
|
|
NextLine(const char *pc)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
while (*pc != 0)
|
|
|
|
{
|
|
|
|
if (pc[0] == '\n' && pc[1] == '\r') return pc + 2;
|
|
|
|
else if (pc[0] == '\n') return pc + 1;
|
|
|
|
pc++;
|
|
|
|
}
|
|
|
|
return pc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2019-08-18 11:00:42 +00:00
|
|
|
TokenLength(const char *pc)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
int length = 0;
|
|
|
|
|
|
|
|
while (!IsSeparator(*pc++)) length++;
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *
|
|
|
|
NextToken(const char *pc)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
/* Skip token */
|
|
|
|
while (!IsSeparator(*pc)) pc++;
|
|
|
|
|
|
|
|
/* Skip white spaces */
|
|
|
|
while (*pc == ' ' || *pc == '\t') pc++;
|
|
|
|
|
|
|
|
/* Check for end of line */
|
|
|
|
if (*pc == '\n' || *pc == '\r' || *pc == 0) return 0;
|
|
|
|
|
|
|
|
/* Check for comment */
|
|
|
|
if (*pc == '#' || *pc == ';') return 0;
|
|
|
|
|
|
|
|
return pc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OutputHeader_stub(FILE *file)
|
|
|
|
{
|
|
|
|
fprintf(file, "/* This file is autogenerated, do not edit. */\n\n"
|
2014-09-26 12:43:12 +00:00
|
|
|
"#include <stubs.h>\n");
|
|
|
|
|
|
|
|
if (gbTracing)
|
|
|
|
{
|
|
|
|
fprintf(file, "#include <wine/debug.h>\n");
|
|
|
|
fprintf(file, "#include <inttypes.h>\n");
|
|
|
|
fprintf(file, "WINE_DECLARE_DEBUG_CHANNEL(relay);\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(file, "\n");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
OutputLine_stub(FILE *file, EXPORT *pexp)
|
|
|
|
{
|
|
|
|
int i;
|
2014-09-26 12:43:12 +00:00
|
|
|
int bRelay = 0;
|
2014-09-27 10:58:07 +00:00
|
|
|
int bInPrototype = 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2021-04-29 13:30:59 +00:00
|
|
|
/* Workaround for forwarded externs. See here for an explanation:
|
2020-11-25 13:26:53 +00:00
|
|
|
* https://stackoverflow.com/questions/4060143/forwarding-data-in-a-dll */
|
2021-04-29 13:30:59 +00:00
|
|
|
if (gbMSComp &&
|
|
|
|
(pexp->nCallingConvention == CC_EXTERN) &&
|
2020-11-25 13:26:53 +00:00
|
|
|
(pexp->strTarget.buf != NULL) &&
|
|
|
|
(!!ScanToken(pexp->strTarget.buf, '.')))
|
|
|
|
{
|
|
|
|
fprintf(file, "#pragma comment(linker,\"/export:%s%.*s=%.*s,DATA\")\n\n",
|
|
|
|
gpszUnderscore, pexp->strName.len, pexp->strName.buf, pexp->strTarget.len, pexp->strTarget.buf);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
if (pexp->nCallingConvention != CC_STUB &&
|
2014-09-26 12:43:12 +00:00
|
|
|
(pexp->uFlags & FL_STUB) == 0)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
/* Only relay trace stdcall C functions */
|
|
|
|
if (!gbTracing || (pexp->nCallingConvention != CC_STDCALL)
|
|
|
|
|| (pexp->uFlags & FL_NORELAY)
|
|
|
|
|| (pexp->strName.buf[0] == '?'))
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
bRelay = 1;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-26 12:43:12 +00:00
|
|
|
/* Declare the "real" function */
|
|
|
|
if (bRelay)
|
2011-07-21 18:39:24 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
fprintf(file, "extern ");
|
|
|
|
bInPrototype = 1;
|
2011-07-21 18:39:24 +00:00
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
|
|
|
|
do
|
2011-07-21 18:39:24 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
if (pexp->uFlags & FL_REGISTER)
|
|
|
|
{
|
|
|
|
/* FIXME: Not sure this is right */
|
|
|
|
fprintf(file, "void ");
|
|
|
|
}
|
|
|
|
else if (pexp->uFlags & FL_RET64)
|
|
|
|
{
|
|
|
|
fprintf(file, "__int64 ");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(file, "int ");
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2014-09-26 12:43:12 +00:00
|
|
|
if ((giArch == ARCH_X86) &&
|
|
|
|
pexp->nCallingConvention == CC_STDCALL)
|
|
|
|
{
|
|
|
|
fprintf(file, "__stdcall ");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for C++ */
|
|
|
|
if (pexp->strName.buf[0] == '?')
|
|
|
|
{
|
|
|
|
fprintf(file, "stub_function%d(", pexp->nNumber);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!bRelay || bInPrototype)
|
|
|
|
fprintf(file, "%.*s(", pexp->strName.len, pexp->strName.buf);
|
|
|
|
else
|
|
|
|
fprintf(file, "$relaytrace$%.*s(", pexp->strName.len, pexp->strName.buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < pexp->nArgCount; i++)
|
|
|
|
{
|
|
|
|
if (i != 0) fprintf(file, ", ");
|
|
|
|
switch (pexp->anArgs[i])
|
|
|
|
{
|
2020-08-29 23:57:00 +00:00
|
|
|
case ARG_LONG: fprintf(file, "long"); break;
|
|
|
|
case ARG_PTR: fprintf(file, "void*"); break;
|
|
|
|
case ARG_STR: fprintf(file, "char*"); break;
|
|
|
|
case ARG_WSTR: fprintf(file, "wchar_t*"); break;
|
|
|
|
case ARG_DBL: fprintf(file, "double"); break;
|
|
|
|
case ARG_INT64: fprintf(file, "__int64"); break;
|
[SPEC2DEF] Improve the 'int128' arguments display. Addendum to commit a28fa3fd.
'int128' arguments are NOT almost always GUID, as was claimed, and the
usage of the wine_dbgstr_guid() function to display them would require
the presence of yet another wine-specific header.
Instead, define a "MyInt128" typedef, local to the stub file being
generated, as the structure of two __int64's (lower and upper), and
print this "MyInt128" as the couple of these two __int64's.
Besides, display the __int64 numbers prefixed with "0x" (together with
the PRIx64 formatter).
Finally, when generating the debug-print function calls, it is useless to
explicitly cast the 'aX' variables with their types, because their types
are already known from the prototype of the stub-function!!
Therefore we can use the same `fprintf(file, "a%d", i);` for all the
ARG_LONG, ARG_PTR, ARG_STR, ARG_WSTR, ARG_DBL, ARG_INT64 and ARG_FLOAT.
Only in the case of ARG_INT128 we output "a%d.lower, a%d.upper" .
2020-08-30 00:02:26 +00:00
|
|
|
/* __int128 is not supported on x86, so use a custom type */
|
|
|
|
case ARG_INT128: fprintf(file, "MyInt128"); break;
|
2020-08-29 23:57:00 +00:00
|
|
|
case ARG_FLOAT: fprintf(file, "float"); break;
|
2014-09-26 12:43:12 +00:00
|
|
|
}
|
|
|
|
fprintf(file, " a%d", i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bInPrototype)
|
|
|
|
{
|
|
|
|
fprintf(file, ");\n\n");
|
|
|
|
}
|
|
|
|
} while (bInPrototype--);
|
|
|
|
|
|
|
|
if (!bRelay)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
fprintf(file, ")\n{\n\tDbgPrint(\"WARNING: calling stub %.*s(",
|
|
|
|
pexp->strName.len, pexp->strName.buf);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(file, ")\n{\n");
|
|
|
|
if (pexp->uFlags & FL_REGISTER)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
/* No return value */
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
else if (pexp->uFlags & FL_RET64)
|
|
|
|
{
|
|
|
|
fprintf(file, "\t__int64 retval;\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(file, "\tint retval;\n");
|
|
|
|
}
|
2014-09-26 12:43:27 +00:00
|
|
|
fprintf(file, "\tif (TRACE_ON(relay))\n\t\tDPRINTF(\"%s: %.*s(",
|
2015-02-04 23:56:23 +00:00
|
|
|
pszDllName, pexp->strName.len, pexp->strName.buf);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < pexp->nArgCount; i++)
|
|
|
|
{
|
|
|
|
if (i != 0) fprintf(file, ",");
|
|
|
|
switch (pexp->anArgs[i])
|
|
|
|
{
|
2020-08-29 23:57:00 +00:00
|
|
|
case ARG_LONG: fprintf(file, "0x%%lx"); break;
|
|
|
|
case ARG_PTR: fprintf(file, "0x%%p"); break;
|
|
|
|
case ARG_STR: fprintf(file, "'%%s'"); break;
|
|
|
|
case ARG_WSTR: fprintf(file, "'%%ws'"); break;
|
|
|
|
case ARG_DBL: fprintf(file, "%%f"); break;
|
[SPEC2DEF] Improve the 'int128' arguments display. Addendum to commit a28fa3fd.
'int128' arguments are NOT almost always GUID, as was claimed, and the
usage of the wine_dbgstr_guid() function to display them would require
the presence of yet another wine-specific header.
Instead, define a "MyInt128" typedef, local to the stub file being
generated, as the structure of two __int64's (lower and upper), and
print this "MyInt128" as the couple of these two __int64's.
Besides, display the __int64 numbers prefixed with "0x" (together with
the PRIx64 formatter).
Finally, when generating the debug-print function calls, it is useless to
explicitly cast the 'aX' variables with their types, because their types
are already known from the prototype of the stub-function!!
Therefore we can use the same `fprintf(file, "a%d", i);` for all the
ARG_LONG, ARG_PTR, ARG_STR, ARG_WSTR, ARG_DBL, ARG_INT64 and ARG_FLOAT.
Only in the case of ARG_INT128 we output "a%d.lower, a%d.upper" .
2020-08-30 00:02:26 +00:00
|
|
|
case ARG_INT64: fprintf(file, "0x%%\"PRIx64\""); break;
|
|
|
|
case ARG_INT128: fprintf(file, "0x%%\"PRIx64\"-0x%%\"PRIx64\""); break;
|
2020-08-29 23:57:00 +00:00
|
|
|
case ARG_FLOAT: fprintf(file, "%%f"); break;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(file, ")\\n\"");
|
|
|
|
|
|
|
|
for (i = 0; i < pexp->nArgCount; i++)
|
|
|
|
{
|
|
|
|
fprintf(file, ", ");
|
|
|
|
switch (pexp->anArgs[i])
|
|
|
|
{
|
[SPEC2DEF] Improve the 'int128' arguments display. Addendum to commit a28fa3fd.
'int128' arguments are NOT almost always GUID, as was claimed, and the
usage of the wine_dbgstr_guid() function to display them would require
the presence of yet another wine-specific header.
Instead, define a "MyInt128" typedef, local to the stub file being
generated, as the structure of two __int64's (lower and upper), and
print this "MyInt128" as the couple of these two __int64's.
Besides, display the __int64 numbers prefixed with "0x" (together with
the PRIx64 formatter).
Finally, when generating the debug-print function calls, it is useless to
explicitly cast the 'aX' variables with their types, because their types
are already known from the prototype of the stub-function!!
Therefore we can use the same `fprintf(file, "a%d", i);` for all the
ARG_LONG, ARG_PTR, ARG_STR, ARG_WSTR, ARG_DBL, ARG_INT64 and ARG_FLOAT.
Only in the case of ARG_INT128 we output "a%d.lower, a%d.upper" .
2020-08-30 00:02:26 +00:00
|
|
|
case ARG_LONG: case ARG_PTR: case ARG_STR:
|
|
|
|
case ARG_WSTR: case ARG_DBL: case ARG_INT64:
|
|
|
|
fprintf(file, "a%d", i); break;
|
|
|
|
case ARG_INT128:
|
|
|
|
fprintf(file, "a%d.lower, a%d.upper", i, i); break;
|
|
|
|
case ARG_FLOAT:
|
|
|
|
fprintf(file, "a%d", i); break;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(file, ");\n");
|
|
|
|
|
|
|
|
if (pexp->nCallingConvention == CC_STUB)
|
|
|
|
{
|
|
|
|
fprintf(file, "\t__wine_spec_unimplemented_stub(\"%s\", __FUNCTION__);\n", pszDllName);
|
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
else if (bRelay)
|
|
|
|
{
|
|
|
|
if (pexp->uFlags & FL_REGISTER)
|
|
|
|
{
|
|
|
|
fprintf(file,"\t");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(file, "\tretval = ");
|
|
|
|
}
|
|
|
|
fprintf(file, "%.*s(", pexp->strName.len, pexp->strName.buf);
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2014-09-26 12:43:12 +00:00
|
|
|
for (i = 0; i < pexp->nArgCount; i++)
|
|
|
|
{
|
|
|
|
if (i != 0) fprintf(file, ", ");
|
|
|
|
fprintf(file, "a%d", i);
|
|
|
|
}
|
|
|
|
fprintf(file, ");\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bRelay)
|
|
|
|
fprintf(file, "\treturn 0;\n}\n\n");
|
|
|
|
else if ((pexp->uFlags & FL_REGISTER) == 0)
|
|
|
|
{
|
|
|
|
if (pexp->uFlags & FL_RET64)
|
|
|
|
{
|
[SPEC2DEF] Improve the 'int128' arguments display. Addendum to commit a28fa3fd.
'int128' arguments are NOT almost always GUID, as was claimed, and the
usage of the wine_dbgstr_guid() function to display them would require
the presence of yet another wine-specific header.
Instead, define a "MyInt128" typedef, local to the stub file being
generated, as the structure of two __int64's (lower and upper), and
print this "MyInt128" as the couple of these two __int64's.
Besides, display the __int64 numbers prefixed with "0x" (together with
the PRIx64 formatter).
Finally, when generating the debug-print function calls, it is useless to
explicitly cast the 'aX' variables with their types, because their types
are already known from the prototype of the stub-function!!
Therefore we can use the same `fprintf(file, "a%d", i);` for all the
ARG_LONG, ARG_PTR, ARG_STR, ARG_WSTR, ARG_DBL, ARG_INT64 and ARG_FLOAT.
Only in the case of ARG_INT128 we output "a%d.lower, a%d.upper" .
2020-08-30 00:02:26 +00:00
|
|
|
fprintf(file, "\tif (TRACE_ON(relay))\n\t\tDPRINTF(\"%s: %.*s: retval = 0x%%\"PRIx64\"\\n\", retval);\n",
|
2015-02-04 23:56:23 +00:00
|
|
|
pszDllName, pexp->strName.len, pexp->strName.buf);
|
2014-09-26 12:43:12 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-09-26 12:43:27 +00:00
|
|
|
fprintf(file, "\tif (TRACE_ON(relay))\n\t\tDPRINTF(\"%s: %.*s: retval = 0x%%lx\\n\", retval);\n",
|
2015-02-04 23:56:23 +00:00
|
|
|
pszDllName, pexp->strName.len, pexp->strName.buf);
|
2014-09-26 12:43:12 +00:00
|
|
|
}
|
|
|
|
fprintf(file, "\treturn retval;\n}\n\n");
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OutputHeader_asmstub(FILE *file, char *libname)
|
|
|
|
{
|
2021-09-12 17:49:54 +00:00
|
|
|
fprintf(file, "; File generated automatically, do not edit!\n\n");
|
2011-06-06 09:58:58 +00:00
|
|
|
|
|
|
|
if (giArch == ARCH_X86)
|
2014-05-30 00:30:38 +00:00
|
|
|
{
|
2014-09-05 20:07:53 +00:00
|
|
|
fprintf(file, ".586\n.model flat\n.code\n");
|
|
|
|
}
|
|
|
|
else if (giArch == ARCH_AMD64)
|
|
|
|
{
|
|
|
|
fprintf(file, ".code\n");
|
2014-05-30 00:30:38 +00:00
|
|
|
}
|
2021-10-22 15:52:32 +00:00
|
|
|
else if (giArch == ARCH_ARM || giArch == ARCH_ARM64)
|
2014-05-30 00:30:38 +00:00
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
fprintf(file, " AREA |.text|,ALIGN=2,CODE,READONLY\n\n");
|
2014-05-30 00:30:38 +00:00
|
|
|
}
|
2014-09-05 20:07:53 +00:00
|
|
|
}
|
2011-06-06 09:58:58 +00:00
|
|
|
|
2014-09-05 20:07:53 +00:00
|
|
|
void
|
2014-09-29 17:43:32 +00:00
|
|
|
Output_stublabel(FILE *fileDest, char* pszSymbolName)
|
2014-09-05 20:07:53 +00:00
|
|
|
{
|
2021-10-22 15:52:32 +00:00
|
|
|
if (giArch == ARCH_ARM || giArch == ARCH_ARM64)
|
2014-09-05 20:07:53 +00:00
|
|
|
{
|
|
|
|
fprintf(fileDest,
|
2015-01-20 18:33:33 +00:00
|
|
|
"\tEXPORT |%s| [FUNC]\n|%s|\n",
|
2014-09-05 20:07:53 +00:00
|
|
|
pszSymbolName,
|
|
|
|
pszSymbolName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(fileDest,
|
|
|
|
"PUBLIC %s\n%s: nop\n",
|
|
|
|
pszSymbolName,
|
|
|
|
pszSymbolName);
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
OutputLine_asmstub(FILE *fileDest, EXPORT *pexp)
|
|
|
|
{
|
2014-09-05 20:07:53 +00:00
|
|
|
char szNameBuffer[128];
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
/* Handle autoname */
|
2013-11-17 22:11:04 +00:00
|
|
|
if (pexp->strName.len == 1 && pexp->strName.buf[0] == '@')
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2019-08-29 16:15:14 +00:00
|
|
|
sprintf(szNameBuffer, "%s_stub_ordinal%d",
|
|
|
|
gpszUnderscore, pexp->nOrdinal);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2011-06-26 22:23:08 +00:00
|
|
|
else if (giArch != ARCH_X86)
|
2011-06-06 09:58:58 +00:00
|
|
|
{
|
2019-08-29 16:16:06 +00:00
|
|
|
/* Does the string already have stdcall decoration? */
|
|
|
|
const char *pcAt = ScanToken(pexp->strName.buf, '@');
|
2020-09-21 20:39:24 +00:00
|
|
|
if (pcAt && (pcAt < (pexp->strName.buf + pexp->strName.len)) &&
|
2019-08-29 16:16:06 +00:00
|
|
|
(pexp->strName.buf[0] == '_'))
|
|
|
|
{
|
|
|
|
/* Skip leading underscore and remove trailing decoration */
|
|
|
|
sprintf(szNameBuffer, "_stub_%.*s",
|
|
|
|
(int)(pcAt - pexp->strName.buf - 1),
|
|
|
|
pexp->strName.buf + 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sprintf(szNameBuffer, "_stub_%.*s",
|
|
|
|
pexp->strName.len, pexp->strName.buf);
|
|
|
|
}
|
2011-06-06 09:58:58 +00:00
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
else if (pexp->nCallingConvention == CC_STDCALL)
|
|
|
|
{
|
2014-09-05 20:07:53 +00:00
|
|
|
sprintf(szNameBuffer, "__stub_%.*s@%d",
|
2013-11-17 22:11:04 +00:00
|
|
|
pexp->strName.len, pexp->strName.buf, pexp->nStackBytes);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else if (pexp->nCallingConvention == CC_FASTCALL)
|
|
|
|
{
|
2014-09-05 20:07:53 +00:00
|
|
|
sprintf(szNameBuffer, "@_stub_%.*s@%d",
|
2013-11-17 22:11:04 +00:00
|
|
|
pexp->strName.len, pexp->strName.buf, pexp->nStackBytes);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2014-09-29 17:43:32 +00:00
|
|
|
else if ((pexp->nCallingConvention == CC_CDECL) ||
|
|
|
|
(pexp->nCallingConvention == CC_THISCALL) ||
|
|
|
|
(pexp->nCallingConvention == CC_EXTERN) ||
|
|
|
|
(pexp->nCallingConvention == CC_STUB))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-05 20:07:53 +00:00
|
|
|
sprintf(szNameBuffer, "__stub_%.*s",
|
2013-11-17 22:11:04 +00:00
|
|
|
pexp->strName.len, pexp->strName.buf);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2014-09-29 17:43:32 +00:00
|
|
|
else
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-29 17:43:32 +00:00
|
|
|
fprintf(stderr, "Invalid calling convention");
|
|
|
|
return 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-29 17:43:32 +00:00
|
|
|
Output_stublabel(fileDest, szNameBuffer);
|
2014-09-05 20:07:53 +00:00
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OutputHeader_def(FILE *file, char *libname)
|
|
|
|
{
|
|
|
|
fprintf(file,
|
|
|
|
"; File generated automatically, do not edit!\n\n"
|
2013-08-10 15:00:50 +00:00
|
|
|
"NAME %s\n\n"
|
2011-05-16 13:12:07 +00:00
|
|
|
"EXPORTS\n",
|
|
|
|
libname);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-11-17 22:11:04 +00:00
|
|
|
PrintName(FILE *fileDest, EXPORT *pexp, PSTRING pstr, int fDeco)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
const char *pcName = pstr->buf;
|
|
|
|
int nNameLength = pstr->len;
|
|
|
|
const char* pcDot, *pcAt;
|
2019-08-18 13:32:22 +00:00
|
|
|
char namebuffer[19];
|
2019-08-29 16:15:14 +00:00
|
|
|
|
|
|
|
if ((nNameLength == 1) && (pcName[0] == '@'))
|
|
|
|
{
|
|
|
|
sprintf(namebuffer, "ordinal%d", pexp->nOrdinal);
|
|
|
|
pcName = namebuffer;
|
|
|
|
nNameLength = strlen(namebuffer);
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2014-05-10 14:12:20 +00:00
|
|
|
/* Check for non-x86 first */
|
|
|
|
if (giArch != ARCH_X86)
|
|
|
|
{
|
|
|
|
/* Does the string already have stdcall decoration? */
|
|
|
|
pcAt = ScanToken(pcName, '@');
|
2014-05-17 10:58:10 +00:00
|
|
|
if (pcAt && (pcAt < (pcName + nNameLength)) && (pcName[0] == '_'))
|
2014-05-10 14:12:20 +00:00
|
|
|
{
|
|
|
|
/* Skip leading underscore and remove trailing decoration */
|
|
|
|
pcName++;
|
2019-04-15 11:29:33 +00:00
|
|
|
nNameLength = (int)(pcAt - pcName);
|
2014-05-10 14:12:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Print the undecorated function name */
|
|
|
|
fprintf(fileDest, "%.*s", nNameLength, pcName);
|
|
|
|
}
|
|
|
|
else if (fDeco &&
|
2013-11-17 22:11:04 +00:00
|
|
|
((pexp->nCallingConvention == CC_STDCALL) ||
|
|
|
|
(pexp->nCallingConvention == CC_FASTCALL)))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2021-04-29 13:30:59 +00:00
|
|
|
/* Beware with C++ exports */
|
|
|
|
int is_cpp = pcName[0] == '?';
|
|
|
|
|
2013-11-17 22:11:04 +00:00
|
|
|
/* Scan for a dll forwarding dot */
|
|
|
|
pcDot = ScanToken(pcName, '.');
|
|
|
|
if (pcDot)
|
|
|
|
{
|
|
|
|
/* First print the dll name, followed by a dot */
|
2019-04-15 11:29:33 +00:00
|
|
|
nNameLength = (int)(pcDot - pcName);
|
2013-11-17 22:11:04 +00:00
|
|
|
fprintf(fileDest, "%.*s.", nNameLength, pcName);
|
|
|
|
|
|
|
|
/* Now the actual function name */
|
|
|
|
pcName = pcDot + 1;
|
|
|
|
nNameLength = pexp->strTarget.len - nNameLength - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Does the string already have decoration? */
|
|
|
|
pcAt = ScanToken(pcName, '@');
|
|
|
|
if (pcAt && (pcAt < (pcName + nNameLength)))
|
|
|
|
{
|
2021-04-29 13:30:59 +00:00
|
|
|
/* On GCC, we need to remove the leading stdcall underscore, but not for C++ exports */
|
|
|
|
if (!gbMSComp && !is_cpp && (pexp->nCallingConvention == CC_STDCALL))
|
2013-11-17 22:11:04 +00:00
|
|
|
{
|
|
|
|
pcName++;
|
|
|
|
nNameLength--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Print the already decorated function name */
|
|
|
|
fprintf(fileDest, "%.*s", nNameLength, pcName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Print the prefix, but skip it for (GCC && stdcall) */
|
|
|
|
if (gbMSComp || (pexp->nCallingConvention != CC_STDCALL))
|
|
|
|
{
|
2013-11-17 22:43:23 +00:00
|
|
|
fprintf(fileDest, "%c", pexp->nCallingConvention == CC_FASTCALL ? '@' : '_');
|
2013-11-17 22:11:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Print the name with trailing decoration */
|
|
|
|
fprintf(fileDest, "%.*s@%d", nNameLength, pcName, pexp->nStackBytes);
|
|
|
|
}
|
2011-06-26 22:23:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
/* Print the undecorated function name */
|
|
|
|
fprintf(fileDest, "%.*s", nNameLength, pcName);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-17 22:11:04 +00:00
|
|
|
void
|
|
|
|
OutputLine_def_MS(FILE *fileDest, EXPORT *pexp)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
PrintName(fileDest, pexp, &pexp->strName, 0);
|
2011-06-26 22:23:08 +00:00
|
|
|
|
|
|
|
if (gbImportLib)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
/* Redirect to a stub function, to get the right decoration in the lib */
|
2019-08-29 16:15:14 +00:00
|
|
|
fprintf(fileDest, "=_stub_");
|
|
|
|
PrintName(fileDest, pexp, &pexp->strName, 0);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2013-11-17 22:11:04 +00:00
|
|
|
else if (pexp->strTarget.buf)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
if (pexp->strName.buf[0] == '?')
|
2013-09-15 14:33:33 +00:00
|
|
|
{
|
2015-02-14 14:28:02 +00:00
|
|
|
//fprintf(stderr, "warning: ignoring C++ redirection %.*s -> %.*s\n",
|
|
|
|
// pexp->strName.len, pexp->strName.buf, pexp->strTarget.len, pexp->strTarget.buf);
|
2013-09-15 14:33:33 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(fileDest, "=");
|
2013-11-17 22:11:04 +00:00
|
|
|
|
|
|
|
/* If the original name was decorated, use decoration in the forwarder as well */
|
|
|
|
if ((giArch == ARCH_X86) && ScanToken(pexp->strName.buf, '@') &&
|
|
|
|
!ScanToken(pexp->strTarget.buf, '@') &&
|
|
|
|
((pexp->nCallingConvention == CC_STDCALL) ||
|
|
|
|
(pexp->nCallingConvention == CC_FASTCALL)) )
|
|
|
|
{
|
|
|
|
PrintName(fileDest, pexp, &pexp->strTarget, 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Write the undecorated redirection name */
|
|
|
|
fprintf(fileDest, "%.*s", pexp->strTarget.len, pexp->strTarget.buf);
|
|
|
|
}
|
2013-09-15 14:33:33 +00:00
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2011-07-21 18:39:24 +00:00
|
|
|
else if (((pexp->uFlags & FL_STUB) || (pexp->nCallingConvention == CC_STUB)) &&
|
2013-11-17 22:11:04 +00:00
|
|
|
(pexp->strName.buf[0] == '?'))
|
2011-07-21 18:39:24 +00:00
|
|
|
{
|
|
|
|
/* C++ stubs are forwarded to C stubs */
|
2013-11-17 22:11:04 +00:00
|
|
|
fprintf(fileDest, "=stub_function%d", pexp->nNumber);
|
2011-07-21 18:39:24 +00:00
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
else if (gbTracing && ((pexp->uFlags & FL_NORELAY) == 0) && (pexp->nCallingConvention == CC_STDCALL) &&
|
|
|
|
(pexp->strName.buf[0] != '?'))
|
|
|
|
{
|
|
|
|
/* Redirect it to the relay-tracing trampoline */
|
|
|
|
fprintf(fileDest, "=$relaytrace$%.*s", pexp->strName.len, pexp->strName.buf);
|
|
|
|
}
|
2013-11-17 22:11:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OutputLine_def_GCC(FILE *fileDest, EXPORT *pexp)
|
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
int bTracing = 0;
|
2013-11-17 22:11:04 +00:00
|
|
|
/* Print the function name, with decoration for export libs */
|
|
|
|
PrintName(fileDest, pexp, &pexp->strName, gbImportLib);
|
|
|
|
DbgPrint("Generating def line for '%.*s'\n", pexp->strName.len, pexp->strName.buf);
|
|
|
|
|
|
|
|
/* Check if this is a forwarded export */
|
|
|
|
if (pexp->strTarget.buf)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
int fIsExternal = !!ScanToken(pexp->strTarget.buf, '.');
|
2013-11-17 22:31:15 +00:00
|
|
|
DbgPrint("Got redirect '%.*s'\n", pexp->strTarget.len, pexp->strTarget.buf);
|
2013-11-17 22:11:04 +00:00
|
|
|
|
|
|
|
/* print the target name, don't decorate if it is external */
|
2011-05-16 13:12:07 +00:00
|
|
|
fprintf(fileDest, "=");
|
2013-11-17 22:11:04 +00:00
|
|
|
PrintName(fileDest, pexp, &pexp->strTarget, !fIsExternal);
|
|
|
|
}
|
|
|
|
else if (((pexp->uFlags & FL_STUB) || (pexp->nCallingConvention == CC_STUB)) &&
|
|
|
|
(pexp->strName.buf[0] == '?'))
|
|
|
|
{
|
|
|
|
/* C++ stubs are forwarded to C stubs */
|
|
|
|
fprintf(fileDest, "=stub_function%d", pexp->nNumber);
|
|
|
|
}
|
2014-09-30 21:13:57 +00:00
|
|
|
else if (gbTracing && ((pexp->uFlags & FL_NORELAY) == 0) &&
|
|
|
|
(pexp->nCallingConvention == CC_STDCALL) &&
|
2014-09-26 12:43:12 +00:00
|
|
|
(pexp->strName.buf[0] != '?'))
|
|
|
|
{
|
|
|
|
/* Redirect it to the relay-tracing trampoline */
|
|
|
|
char buf[256];
|
|
|
|
STRING strTarget;
|
|
|
|
fprintf(fileDest, "=");
|
|
|
|
sprintf(buf, "$relaytrace$%.*s", pexp->strName.len, pexp->strName.buf);
|
|
|
|
strTarget.buf = buf;
|
|
|
|
strTarget.len = pexp->strName.len + 12;
|
|
|
|
PrintName(fileDest, pexp, &strTarget, 1);
|
|
|
|
bTracing = 1;
|
|
|
|
}
|
2013-11-17 22:11:04 +00:00
|
|
|
|
2021-04-29 13:30:59 +00:00
|
|
|
/* Special handling for stdcall and fastcall, but not C++ exports*/
|
2013-11-17 22:11:04 +00:00
|
|
|
if ((giArch == ARCH_X86) &&
|
2021-04-29 13:30:59 +00:00
|
|
|
(pexp->strName.buf[0] != '?') &&
|
2013-11-17 22:11:04 +00:00
|
|
|
((pexp->nCallingConvention == CC_STDCALL) ||
|
|
|
|
(pexp->nCallingConvention == CC_FASTCALL)))
|
|
|
|
{
|
|
|
|
/* Is this the import lib? */
|
|
|
|
if (gbImportLib)
|
|
|
|
{
|
|
|
|
/* Is the name in the spec file decorated? */
|
|
|
|
const char* pcDeco = ScanToken(pexp->strName.buf, '@');
|
2020-09-21 20:39:24 +00:00
|
|
|
if (pcDeco &&
|
2019-08-29 16:16:06 +00:00
|
|
|
(pexp->strName.len > 1) &&
|
|
|
|
(pcDeco < pexp->strName.buf + pexp->strName.len))
|
2013-11-17 22:11:04 +00:00
|
|
|
{
|
|
|
|
/* Write the name including the leading @ */
|
|
|
|
fprintf(fileDest, "==%.*s", pexp->strName.len, pexp->strName.buf);
|
|
|
|
}
|
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
else if ((!pexp->strTarget.buf) && !(bTracing))
|
2013-11-17 22:11:04 +00:00
|
|
|
{
|
|
|
|
/* Write a forwarder to the actual decorated symbol */
|
|
|
|
fprintf(fileDest, "=");
|
|
|
|
PrintName(fileDest, pexp, &pexp->strName, 1);
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2013-11-17 22:11:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
OutputLine_def(FILE *fileDest, EXPORT *pexp)
|
|
|
|
{
|
2020-10-23 08:31:50 +00:00
|
|
|
/* Don't add private exports to the import lib */
|
|
|
|
if (gbImportLib && (pexp->uFlags & FL_PRIVATE))
|
|
|
|
{
|
|
|
|
DbgPrint("OutputLine_def: skipping private export '%.*s'...\n", pexp->strName.len, pexp->strName.buf);
|
|
|
|
return 1;
|
|
|
|
}
|
2021-04-29 13:30:59 +00:00
|
|
|
|
2020-11-25 13:26:53 +00:00
|
|
|
/* For MS linker, forwarded externs are managed via #pragma comment(linker,"/export:_data=org.data,DATA") */
|
2021-04-29 13:30:59 +00:00
|
|
|
if (gbMSComp && !gbImportLib && (pexp->nCallingConvention == CC_EXTERN) &&
|
2020-11-25 13:26:53 +00:00
|
|
|
(pexp->strTarget.buf != NULL) && !!ScanToken(pexp->strTarget.buf, '.'))
|
|
|
|
{
|
|
|
|
DbgPrint("OutputLine_def: skipping forwarded extern export '%.*s' ->'%.*s'...\n",
|
|
|
|
pexp->strName.len, pexp->strName.buf, pexp->strTarget.len, pexp->strTarget.buf);
|
|
|
|
return 1;
|
|
|
|
}
|
2020-10-23 08:31:50 +00:00
|
|
|
|
2014-05-17 10:58:10 +00:00
|
|
|
DbgPrint("OutputLine_def: '%.*s'...\n", pexp->strName.len, pexp->strName.buf);
|
2013-11-17 22:11:04 +00:00
|
|
|
fprintf(fileDest, " ");
|
|
|
|
|
|
|
|
if (gbMSComp)
|
|
|
|
OutputLine_def_MS(fileDest, pexp);
|
|
|
|
else
|
|
|
|
OutputLine_def_GCC(fileDest, pexp);
|
2011-05-16 13:12:07 +00:00
|
|
|
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
/* On GCC builds we force ordinals */
|
|
|
|
if ((pexp->uFlags & FL_ORDINAL) || (!gbMSComp && !gbImportLib))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
fprintf(fileDest, " @%d", pexp->nOrdinal);
|
|
|
|
}
|
|
|
|
|
2013-12-15 22:01:53 +00:00
|
|
|
if (pexp->uFlags & FL_NONAME)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-12-15 22:01:53 +00:00
|
|
|
fprintf(fileDest, " NONAME");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2015-12-28 14:42:41 +00:00
|
|
|
/* Either PRIVATE or DATA */
|
2011-05-16 13:12:07 +00:00
|
|
|
if (pexp->uFlags & FL_PRIVATE)
|
|
|
|
{
|
|
|
|
fprintf(fileDest, " PRIVATE");
|
|
|
|
}
|
2015-12-28 14:42:41 +00:00
|
|
|
else if (pexp->nCallingConvention == CC_EXTERN)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-12-15 22:01:53 +00:00
|
|
|
fprintf(fileDest, " DATA");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(fileDest, "\n");
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-08-17 18:32:15 +00:00
|
|
|
void
|
|
|
|
Fatalv(
|
|
|
|
const char* filename,
|
|
|
|
unsigned nLine,
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *pcLine,
|
|
|
|
const char *pc,
|
2019-08-17 18:32:15 +00:00
|
|
|
size_t errorlen,
|
|
|
|
const char *format,
|
|
|
|
va_list argptr)
|
|
|
|
{
|
|
|
|
unsigned i, errorpos, len;
|
|
|
|
const char* pcLineEnd;
|
|
|
|
|
|
|
|
/* Get the length of the line */
|
|
|
|
pcLineEnd = strpbrk(pcLine, "\r\n");
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
len = (unsigned)(pcLineEnd - pcLine);
|
2019-08-17 18:32:15 +00:00
|
|
|
|
|
|
|
if (pc == NULL)
|
|
|
|
{
|
|
|
|
pc = pcLine + len - 1;
|
|
|
|
errorlen = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
errorpos = (unsigned)(pc - pcLine);
|
|
|
|
|
|
|
|
/* Output the error message */
|
|
|
|
fprintf(stderr, "ERROR: (%s:%u:%u): ", filename, nLine, errorpos);
|
|
|
|
vfprintf(stderr, format, argptr);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
|
|
|
|
/* Output the line with the error */
|
|
|
|
fprintf(stderr, "> %.*s\n", len, pcLine);
|
|
|
|
|
|
|
|
if (errorlen == 0)
|
|
|
|
{
|
|
|
|
errorlen = TokenLength(pc);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < errorpos + 2; i++)
|
|
|
|
{
|
|
|
|
fprintf(stderr, " ");
|
|
|
|
}
|
|
|
|
for (i = 0; i < errorlen; i++)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "~");
|
|
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Fatal(
|
|
|
|
const char* filename,
|
|
|
|
unsigned nLine,
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *pcLine,
|
|
|
|
const char *pc,
|
2019-08-17 18:32:15 +00:00
|
|
|
size_t errorlen,
|
|
|
|
const char *format,
|
|
|
|
...)
|
|
|
|
{
|
|
|
|
va_list argptr;
|
|
|
|
|
|
|
|
va_start(argptr, format);
|
|
|
|
Fatalv(filename, nLine, pcLine, pc, errorlen, format, argptr);
|
|
|
|
va_end(argptr);
|
|
|
|
}
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
EXPORT *
|
|
|
|
ParseFile(char* pcStart, FILE *fileDest, unsigned *cExports)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2019-08-18 13:32:22 +00:00
|
|
|
EXPORT *pexports;
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *pc, *pcLine;
|
2019-08-18 13:32:22 +00:00
|
|
|
int cLines, nLine;
|
2011-05-16 13:12:07 +00:00
|
|
|
EXPORT exp;
|
2019-08-18 13:32:22 +00:00
|
|
|
int included;
|
2015-02-04 23:56:23 +00:00
|
|
|
unsigned int i;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
*cExports = 0;
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
//fprintf(stderr, "info: line %d, pcStart:'%.30s'\n", nLine, pcStart);
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
/* Count the lines */
|
|
|
|
for (cLines = 1, pcLine = pcStart; *pcLine; pcLine = NextLine(pcLine), cLines++)
|
|
|
|
{
|
|
|
|
/* Nothing */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate an array of EXPORT structures */
|
|
|
|
pexports = malloc(cLines * sizeof(EXPORT));
|
|
|
|
if (pexports == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: %s: failed to allocate EXPORT array of %u elements\n", pszSourceFileName, cLines);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
/* Loop all lines */
|
|
|
|
nLine = 1;
|
2011-07-21 18:39:24 +00:00
|
|
|
exp.nNumber = 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
for (pcLine = pcStart; *pcLine; pcLine = NextLine(pcLine), nLine++)
|
|
|
|
{
|
|
|
|
pc = pcLine;
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
exp.strName.buf = NULL;
|
|
|
|
exp.strName.len = 0;
|
|
|
|
exp.strTarget.buf = NULL;
|
|
|
|
exp.strTarget.len = 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
exp.nArgCount = 0;
|
|
|
|
exp.uFlags = 0;
|
2011-07-21 18:39:24 +00:00
|
|
|
exp.nNumber++;
|
2019-08-18 13:32:22 +00:00
|
|
|
exp.nStartVersion = 0;
|
|
|
|
exp.nEndVersion = 0xFFFFFFFF;
|
|
|
|
exp.bVersionIncluded = 1;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Skip white spaces */
|
|
|
|
while (*pc == ' ' || *pc == '\t') pc++;
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
/* Check for line break or comment */
|
|
|
|
if ((*pc == '\r') || (*pc == '\n') ||
|
|
|
|
(*pc == ';') || (*pc == '#'))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* On EOF we are done */
|
|
|
|
if (*pc == 0)
|
|
|
|
{
|
|
|
|
return pexports;
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Now we should get either an ordinal or @ */
|
2014-02-18 20:06:50 +00:00
|
|
|
if (*pc == '@')
|
2019-08-17 18:32:15 +00:00
|
|
|
{
|
2014-02-18 20:06:50 +00:00
|
|
|
exp.nOrdinal = -1;
|
2019-08-17 18:32:15 +00:00
|
|
|
}
|
|
|
|
else if ((*pc >= '0') && (*pc <= '9'))
|
2014-02-18 20:06:50 +00:00
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
char* end;
|
|
|
|
long int number = strtol(pc, &end, 10);
|
|
|
|
if ((*end != ' ') && (*end != '\t'))
|
|
|
|
{
|
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, end, 0, "Unexpected character(s) after ordinal");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((number < 0) || (number > 0xFFFE))
|
|
|
|
{
|
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 0, "Invalid value for ordinal");
|
|
|
|
}
|
|
|
|
|
|
|
|
exp.nOrdinal = number;
|
|
|
|
|
2014-06-29 11:32:49 +00:00
|
|
|
/* The import lib should contain the ordinal only if -ordinal was specified */
|
|
|
|
if (!gbImportLib)
|
|
|
|
exp.uFlags |= FL_ORDINAL;
|
2014-02-18 20:06:50 +00:00
|
|
|
}
|
2019-08-17 18:32:15 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 0, "Expected '@' or ordinal");
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Go to next token (type) */
|
|
|
|
if (!(pc = NextToken(pc)))
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 1, "Unexpected end of line");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2013-11-17 22:11:04 +00:00
|
|
|
//fprintf(stderr, "info: Token:'%.*s'\n", TokenLength(pc), pc);
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Now we should get the type */
|
|
|
|
if (CompareToken(pc, "stdcall"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_STDCALL;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "cdecl") ||
|
|
|
|
CompareToken(pc, "varargs"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_CDECL;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "fastcall"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_FASTCALL;
|
|
|
|
}
|
2011-10-06 11:10:54 +00:00
|
|
|
else if (CompareToken(pc, "thiscall"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_THISCALL;
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
else if (CompareToken(pc, "extern"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_EXTERN;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "stub"))
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_STUB;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 0, "Invalid calling convention");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Go to next token (options or name) */
|
|
|
|
if (!(pc = NextToken(pc)))
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 1, "Unexpected end of line");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle options */
|
|
|
|
included = 1;
|
|
|
|
while (*pc == '-')
|
|
|
|
{
|
2018-02-18 20:22:52 +00:00
|
|
|
if (CompareToken(pc, "-arch="))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
/* Default to not included */
|
|
|
|
included = 0;
|
|
|
|
pc += 5;
|
|
|
|
|
|
|
|
/* Look if we are included */
|
2018-02-18 20:22:52 +00:00
|
|
|
do
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
pc++;
|
|
|
|
if (CompareToken(pc, pszArchString) ||
|
|
|
|
CompareToken(pc, pszArchString2))
|
|
|
|
{
|
|
|
|
included = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip to next arch or end */
|
|
|
|
while (*pc > ',') pc++;
|
2018-02-18 20:22:52 +00:00
|
|
|
} while (*pc == ',');
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "-i386"))
|
|
|
|
{
|
2011-06-06 16:26:46 +00:00
|
|
|
if (giArch != ARCH_X86) included = 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2018-02-18 20:22:52 +00:00
|
|
|
else if (CompareToken(pc, "-version="))
|
|
|
|
{
|
2019-08-18 11:00:42 +00:00
|
|
|
const char *pcVersionStart = pc + 9;
|
2019-08-17 18:32:15 +00:00
|
|
|
|
2018-02-18 20:22:52 +00:00
|
|
|
/* Default to not included */
|
2019-08-18 13:32:22 +00:00
|
|
|
exp.bVersionIncluded = 0;
|
2018-02-18 20:22:52 +00:00
|
|
|
pc += 8;
|
|
|
|
|
|
|
|
/* Look if we are included */
|
|
|
|
do
|
|
|
|
{
|
|
|
|
unsigned version, endversion;
|
|
|
|
|
|
|
|
/* Optionally skip leading '0x' */
|
|
|
|
pc++;
|
|
|
|
if ((pc[0] == '0') && (pc[1] == 'x')) pc += 2;
|
|
|
|
|
|
|
|
/* Now get the version number */
|
2019-08-18 11:00:42 +00:00
|
|
|
endversion = version = strtoul(pc, (char**)&pc, 16);
|
2018-02-18 20:22:52 +00:00
|
|
|
|
|
|
|
/* Check if it's a range */
|
|
|
|
if (pc[0] == '+')
|
|
|
|
{
|
|
|
|
endversion = 0xFFF;
|
|
|
|
pc++;
|
|
|
|
}
|
|
|
|
else if (pc[0] == '-')
|
|
|
|
{
|
|
|
|
/* Optionally skip leading '0x' */
|
|
|
|
pc++;
|
|
|
|
if ((pc[0] == '0') && (pc[1] == 'x')) pc += 2;
|
2019-08-18 11:00:42 +00:00
|
|
|
endversion = strtoul(pc, (char**)&pc, 16);
|
2018-02-18 20:22:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for degenerate range */
|
|
|
|
if (version > endversion)
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName,
|
|
|
|
nLine,
|
|
|
|
pcLine,
|
|
|
|
pcVersionStart,
|
|
|
|
pc - pcVersionStart,
|
|
|
|
"Invalid version range");
|
2018-02-18 20:22:52 +00:00
|
|
|
}
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
exp.nStartVersion = version;
|
|
|
|
exp.nEndVersion = endversion;
|
|
|
|
|
2018-02-18 20:22:52 +00:00
|
|
|
/* Now compare the range with our version */
|
|
|
|
if ((guOsVersion >= version) &&
|
|
|
|
(guOsVersion <= endversion))
|
|
|
|
{
|
2019-08-18 13:32:22 +00:00
|
|
|
exp.bVersionIncluded = 1;
|
2018-02-18 20:22:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip to next arch or end */
|
|
|
|
while (*pc > ',') pc++;
|
|
|
|
|
|
|
|
} while (*pc == ',');
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
else if (CompareToken(pc, "-private"))
|
|
|
|
{
|
|
|
|
exp.uFlags |= FL_PRIVATE;
|
|
|
|
}
|
2014-02-18 20:06:50 +00:00
|
|
|
else if (CompareToken(pc, "-noname"))
|
|
|
|
{
|
|
|
|
exp.uFlags |= FL_ORDINAL | FL_NONAME;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "-ordinal"))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-02-18 20:06:50 +00:00
|
|
|
exp.uFlags |= FL_ORDINAL;
|
2014-08-12 13:21:38 +00:00
|
|
|
/* GCC doesn't automatically import by ordinal if an ordinal
|
|
|
|
* is found in the def file. Force it. */
|
|
|
|
if (gbImportLib && !gbMSComp)
|
|
|
|
exp.uFlags |= FL_NONAME;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "-stub"))
|
|
|
|
{
|
|
|
|
exp.uFlags |= FL_STUB;
|
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
else if (CompareToken(pc, "-norelay"))
|
|
|
|
{
|
|
|
|
exp.uFlags |= FL_NORELAY;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "-ret64"))
|
|
|
|
{
|
|
|
|
exp.uFlags |= FL_RET64;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "-register"))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
exp.uFlags |= FL_REGISTER;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
fprintf(stdout,
|
|
|
|
"INFO: %s line %d: Ignored option: '%.*s'\n",
|
|
|
|
pszSourceFileName,
|
|
|
|
nLine,
|
|
|
|
TokenLength(pc),
|
|
|
|
pc);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Go to next token */
|
|
|
|
pc = NextToken(pc);
|
|
|
|
}
|
|
|
|
|
|
|
|
//fprintf(stderr, "info: Name:'%.10s'\n", pc);
|
|
|
|
|
|
|
|
/* If arch didn't match ours, skip this entry */
|
2019-08-18 13:32:22 +00:00
|
|
|
if (!included) continue;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Get name */
|
2013-11-17 22:11:04 +00:00
|
|
|
exp.strName.buf = pc;
|
|
|
|
exp.strName.len = TokenLength(pc);
|
2019-08-17 18:32:15 +00:00
|
|
|
//DbgPrint("Got name: '%.*s'\n", exp.strName.len, exp.strName.buf);
|
2013-11-17 22:11:04 +00:00
|
|
|
|
|
|
|
/* Check for autoname */
|
|
|
|
if ((exp.strName.len == 1) && (exp.strName.buf[0] == '@'))
|
|
|
|
{
|
2014-02-18 20:06:50 +00:00
|
|
|
exp.uFlags |= FL_ORDINAL | FL_NONAME;
|
2013-11-17 22:11:04 +00:00
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Handle parameters */
|
|
|
|
exp.nStackBytes = 0;
|
2020-09-21 20:39:24 +00:00
|
|
|
pc = NextToken(pc);
|
|
|
|
/* Extern can't have parameters, and it's optional to provide ones for stubs. All other exports must have them */
|
|
|
|
if (!pc && (exp.nCallingConvention != CC_EXTERN && exp.nCallingConvention != CC_STUB))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2020-09-21 20:39:24 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 1, "Unexpected end of line");
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
2020-09-21 20:39:24 +00:00
|
|
|
if (pc && (exp.nCallingConvention != CC_EXTERN))
|
|
|
|
{
|
2011-05-16 13:12:07 +00:00
|
|
|
/* Verify syntax */
|
|
|
|
if (*pc++ != '(')
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc - 1, 0, "Expected '('");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip whitespaces */
|
|
|
|
while (*pc == ' ' || *pc == '\t') pc++;
|
|
|
|
|
|
|
|
exp.nStackBytes = 0;
|
|
|
|
while (*pc >= '0')
|
|
|
|
{
|
|
|
|
if (CompareToken(pc, "long"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 4;
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_LONG;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "double"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 8;
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_DBL;
|
|
|
|
}
|
2014-09-25 18:25:02 +00:00
|
|
|
else if (CompareToken(pc, "ptr"))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
exp.nStackBytes += 4; // sizeof(void*) on x86
|
2014-09-25 18:25:02 +00:00
|
|
|
exp.anArgs[exp.nArgCount] = ARG_PTR;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "str"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 4; // sizeof(void*) on x86
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_STR;
|
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "wstr"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 4; // sizeof(void*) on x86
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_WSTR;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else if (CompareToken(pc, "int64"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 8;
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_INT64;
|
|
|
|
}
|
2012-09-29 22:58:06 +00:00
|
|
|
else if (CompareToken(pc, "int128"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 16;
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_INT128;
|
|
|
|
}
|
2011-10-06 11:10:54 +00:00
|
|
|
else if (CompareToken(pc, "float"))
|
|
|
|
{
|
|
|
|
exp.nStackBytes += 4;
|
|
|
|
exp.anArgs[exp.nArgCount] = ARG_FLOAT;
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
else
|
2019-08-17 18:32:15 +00:00
|
|
|
{
|
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 0, "Unrecognized type");
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
exp.nArgCount++;
|
|
|
|
|
|
|
|
/* Go to next parameter */
|
|
|
|
if (!(pc = NextToken(pc)))
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 1, "Unexpected end of line");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check syntax */
|
|
|
|
if (*pc++ != ')')
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc - 1, 0, "Expected ')'");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2020-09-21 20:39:24 +00:00
|
|
|
|
|
|
|
/* Go to next token */
|
|
|
|
pc = NextToken(pc);
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle special stub cases */
|
|
|
|
if (exp.nCallingConvention == CC_STUB)
|
|
|
|
{
|
2020-09-21 20:39:24 +00:00
|
|
|
/* If we got parameters, assume STDCALL */
|
|
|
|
if (exp.nArgCount != 0)
|
|
|
|
{
|
|
|
|
exp.nCallingConvention = CC_STDCALL;
|
|
|
|
exp.uFlags |= FL_STUB;
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
/* Check for c++ mangled name */
|
2020-09-21 20:39:24 +00:00
|
|
|
if (exp.strName.buf[0] == '?')
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2011-07-21 18:39:24 +00:00
|
|
|
//printf("Found c++ mangled name...\n");
|
2011-05-16 13:12:07 +00:00
|
|
|
//
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Check for stdcall name */
|
2020-09-21 20:39:24 +00:00
|
|
|
const char *p = ScanToken(exp.strName.buf, '@');
|
|
|
|
if (p && (p - exp.strName.buf < exp.strName.len))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
int i;
|
2013-11-17 22:11:04 +00:00
|
|
|
|
|
|
|
/* Truncate the name to before the @ */
|
2020-09-21 20:39:24 +00:00
|
|
|
exp.strName.len = (int)(p - exp.strName.buf);
|
2013-11-17 22:11:04 +00:00
|
|
|
if (exp.strName.len < 1)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, p, 1, "Unexpected @");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
exp.nStackBytes = atoi(p + 1);
|
|
|
|
exp.nArgCount = exp.nStackBytes / 4;
|
|
|
|
exp.nCallingConvention = CC_STDCALL;
|
|
|
|
exp.uFlags |= FL_STUB;
|
|
|
|
for (i = 0; i < exp.nArgCount; i++)
|
|
|
|
exp.anArgs[i] = ARG_LONG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-21 20:39:24 +00:00
|
|
|
/* Check optional redirection */
|
2013-07-20 11:40:04 +00:00
|
|
|
if (pc)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
2013-11-17 22:11:04 +00:00
|
|
|
exp.strTarget.buf = pc;
|
|
|
|
exp.strTarget.len = TokenLength(pc);
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Check syntax (end of line) */
|
|
|
|
if (NextToken(pc))
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, NextToken(pc), 0, "Excess token(s) at end of definition");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
2014-09-26 12:43:12 +00:00
|
|
|
|
|
|
|
/* Don't relay-trace forwarded functions */
|
|
|
|
exp.uFlags |= FL_NORELAY;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
exp.strTarget.buf = NULL;
|
2013-11-17 22:11:04 +00:00
|
|
|
exp.strTarget.len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for no-name without ordinal */
|
2014-02-18 20:06:50 +00:00
|
|
|
if ((exp.uFlags & FL_ORDINAL) && (exp.nOrdinal == -1))
|
2013-11-17 22:11:04 +00:00
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
Fatal(pszSourceFileName, nLine, pcLine, pc, 0, "Ordinal export without ordinal");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 22:11:18 +00:00
|
|
|
/*
|
|
|
|
* Check for special handling of OLE exports, only when MSVC
|
|
|
|
* is not used, since otherwise this is handled by MS LINK.EXE.
|
|
|
|
*/
|
|
|
|
if (!gbMSComp)
|
2015-02-04 23:56:23 +00:00
|
|
|
{
|
2019-06-01 22:11:18 +00:00
|
|
|
/* Check whether the current export is not PRIVATE, or has an ordinal */
|
|
|
|
int bIsNotPrivate = (!gbNotPrivateNoWarn && /*gbImportLib &&*/ !(exp.uFlags & FL_PRIVATE));
|
|
|
|
int bHasOrdinal = (exp.uFlags & FL_ORDINAL);
|
|
|
|
|
|
|
|
/* Check whether the current export is an OLE export, in case any of these tests pass */
|
|
|
|
if (bIsNotPrivate || bHasOrdinal)
|
2015-02-04 23:56:23 +00:00
|
|
|
{
|
2019-06-01 22:11:18 +00:00
|
|
|
for (i = 0; i < ARRAYSIZE(astrOlePrivateExports); ++i)
|
2015-02-04 23:56:23 +00:00
|
|
|
{
|
2019-06-01 22:11:18 +00:00
|
|
|
if (strlen(astrOlePrivateExports[i]) == exp.strName.len &&
|
|
|
|
strncmp(exp.strName.buf, astrOlePrivateExports[i], exp.strName.len) == 0)
|
|
|
|
{
|
|
|
|
/* The current export is an OLE export: display the corresponding warning */
|
|
|
|
if (bIsNotPrivate)
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
fprintf(stderr, "WARNING: %s line %d: Exported symbol '%.*s' should be PRIVATE\n",
|
2019-06-01 22:11:18 +00:00
|
|
|
pszSourceFileName, nLine, exp.strName.len, exp.strName.buf);
|
|
|
|
}
|
|
|
|
if (bHasOrdinal)
|
|
|
|
{
|
2019-08-17 18:32:15 +00:00
|
|
|
fprintf(stderr, "WARNING: %s line %d: exported symbol '%.*s' should not be assigned an ordinal\n",
|
2019-06-01 22:11:18 +00:00
|
|
|
pszSourceFileName, nLine, exp.strName.len, exp.strName.buf);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2015-02-04 23:56:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
pexports[*cExports] = exp;
|
|
|
|
(*cExports)++;
|
2013-11-17 22:11:04 +00:00
|
|
|
gbDebug = 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
return pexports;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
int
|
|
|
|
ApplyOrdinals(EXPORT* pexports, unsigned cExports)
|
|
|
|
{
|
|
|
|
unsigned short i, j;
|
|
|
|
char* used;
|
|
|
|
|
|
|
|
/* Allocate a table to mark used ordinals */
|
|
|
|
used = malloc(65536);
|
|
|
|
if (used == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Failed to allocate memory for ordinal use table\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
memset(used, 0, 65536);
|
|
|
|
|
|
|
|
/* Pass 1: mark the ordinals that are already used */
|
|
|
|
for (i = 0; i < cExports; i++)
|
|
|
|
{
|
|
|
|
if (pexports[i].uFlags & FL_ORDINAL)
|
|
|
|
{
|
|
|
|
if (used[pexports[i].nOrdinal] != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Found duplicate ordinal: %u\n", pexports[i].nOrdinal);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
used[pexports[i].nOrdinal] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Pass 2: apply available ordinals */
|
|
|
|
for (i = 0, j = 1; i < cExports; i++)
|
|
|
|
{
|
2020-03-30 17:41:14 +00:00
|
|
|
if ((pexports[i].uFlags & FL_ORDINAL) == 0 && pexports[i].bVersionIncluded)
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
{
|
|
|
|
while (used[j] != 0)
|
|
|
|
j++;
|
|
|
|
|
|
|
|
pexports[i].nOrdinal = j;
|
|
|
|
used[j] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(used);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
void usage(void)
|
|
|
|
{
|
2014-09-26 12:43:12 +00:00
|
|
|
printf("syntax: spec2def [<options> ...] <spec file>\n"
|
2011-05-16 13:12:07 +00:00
|
|
|
"Possible options:\n"
|
2015-02-04 23:56:23 +00:00
|
|
|
" -h --help print this help screen\n"
|
|
|
|
" -l=<file> generate an asm lib stub\n"
|
|
|
|
" -d=<file> generate a def file\n"
|
|
|
|
" -s=<file> generate a stub file\n"
|
|
|
|
" --ms MSVC compatibility\n"
|
|
|
|
" -n=<name> name of the dll\n"
|
2023-08-05 11:58:17 +00:00
|
|
|
" --version=<version> Sets the version to create exports for\n"
|
2015-02-04 23:56:23 +00:00
|
|
|
" --implib generate a def file for an import library\n"
|
|
|
|
" --no-private-warnings suppress warnings about symbols that should be -private\n"
|
2021-10-22 15:52:32 +00:00
|
|
|
" -a=<arch> set architecture to <arch> (i386, x86_64, arm, arm64)\n"
|
2015-02-04 23:56:23 +00:00
|
|
|
" --with-tracing generate wine-like \"+relay\" trace trampolines (needs -s)\n");
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
size_t nFileSize;
|
2015-02-04 23:56:23 +00:00
|
|
|
char *pszSource, *pszDefFileName = NULL, *pszStubFileName = NULL, *pszLibStubName = NULL;
|
2018-09-05 19:49:02 +00:00
|
|
|
const char* pszVersionOption = "--version=0x";
|
2011-05-16 13:12:07 +00:00
|
|
|
char achDllName[40];
|
|
|
|
FILE *file;
|
2019-08-18 13:32:22 +00:00
|
|
|
unsigned cExports = 0, i;
|
|
|
|
EXPORT *pexports;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Read options */
|
2019-08-18 13:32:22 +00:00
|
|
|
for (i = 1; i < (unsigned)argc && *argv[i] == '-'; i++)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
if ((strcasecmp(argv[i], "--help") == 0) ||
|
|
|
|
(strcasecmp(argv[i], "-h") == 0))
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else if (argv[i][1] == 'd' && argv[i][2] == '=')
|
|
|
|
{
|
|
|
|
pszDefFileName = argv[i] + 3;
|
|
|
|
}
|
|
|
|
else if (argv[i][1] == 'l' && argv[i][2] == '=')
|
|
|
|
{
|
|
|
|
pszLibStubName = argv[i] + 3;
|
|
|
|
}
|
|
|
|
else if (argv[i][1] == 's' && argv[i][2] == '=')
|
|
|
|
{
|
|
|
|
pszStubFileName = argv[i] + 3;
|
|
|
|
}
|
|
|
|
else if (argv[i][1] == 'n' && argv[i][2] == '=')
|
|
|
|
{
|
|
|
|
pszDllName = argv[i] + 3;
|
|
|
|
}
|
2018-09-05 19:49:02 +00:00
|
|
|
else if (strncasecmp(argv[i], pszVersionOption, strlen(pszVersionOption)) == 0)
|
2018-02-18 20:22:52 +00:00
|
|
|
{
|
2018-09-05 19:49:02 +00:00
|
|
|
guOsVersion = strtoul(argv[i] + strlen(pszVersionOption), NULL, 16);
|
2018-02-18 20:22:52 +00:00
|
|
|
}
|
2015-02-04 23:56:23 +00:00
|
|
|
else if (strcasecmp(argv[i], "--implib") == 0)
|
2011-06-26 22:23:08 +00:00
|
|
|
{
|
|
|
|
gbImportLib = 1;
|
|
|
|
}
|
2015-02-04 23:56:23 +00:00
|
|
|
else if (strcasecmp(argv[i], "--ms") == 0)
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
gbMSComp = 1;
|
|
|
|
}
|
2015-02-04 23:56:23 +00:00
|
|
|
else if (strcasecmp(argv[i], "--no-private-warnings") == 0)
|
|
|
|
{
|
|
|
|
gbNotPrivateNoWarn = 1;
|
|
|
|
}
|
|
|
|
else if (strcasecmp(argv[i], "--with-tracing") == 0)
|
2014-09-26 12:43:12 +00:00
|
|
|
{
|
|
|
|
if (!pszStubFileName)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error: cannot use --with-tracing without -s option.\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
gbTracing = 1;
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
else if (argv[i][1] == 'a' && argv[i][2] == '=')
|
|
|
|
{
|
|
|
|
pszArchString = argv[i] + 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Unrecognized option: %s\n", argv[i]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-26 22:23:08 +00:00
|
|
|
if (strcasecmp(pszArchString, "i386") == 0)
|
|
|
|
{
|
|
|
|
giArch = ARCH_X86;
|
|
|
|
gpszUnderscore = "_";
|
|
|
|
}
|
2011-06-06 09:58:58 +00:00
|
|
|
else if (strcasecmp(pszArchString, "x86_64") == 0) giArch = ARCH_AMD64;
|
|
|
|
else if (strcasecmp(pszArchString, "ia64") == 0) giArch = ARCH_IA64;
|
|
|
|
else if (strcasecmp(pszArchString, "arm") == 0) giArch = ARCH_ARM;
|
2021-10-22 15:52:32 +00:00
|
|
|
else if (strcasecmp(pszArchString, "arm64") == 0) giArch = ARCH_ARM64;
|
2011-06-06 09:58:58 +00:00
|
|
|
else if (strcasecmp(pszArchString, "ppc") == 0) giArch = ARCH_PPC;
|
|
|
|
|
2011-06-07 14:19:17 +00:00
|
|
|
if ((giArch == ARCH_AMD64) || (giArch == ARCH_IA64))
|
2011-05-16 13:12:07 +00:00
|
|
|
{
|
|
|
|
pszArchString2 = "win64";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
pszArchString2 = "win32";
|
|
|
|
|
|
|
|
/* Set a default dll name */
|
|
|
|
if (!pszDllName)
|
|
|
|
{
|
|
|
|
char *p1, *p2;
|
2011-06-06 09:58:58 +00:00
|
|
|
size_t len;
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
p1 = strrchr(argv[i], '\\');
|
|
|
|
if (!p1) p1 = strrchr(argv[i], '/');
|
|
|
|
p2 = p1 = p1 ? p1 + 1 : argv[i];
|
|
|
|
|
|
|
|
/* walk up to '.' */
|
|
|
|
while (*p2 != '.' && *p2 != 0) p2++;
|
|
|
|
len = p2 - p1;
|
|
|
|
if (len >= sizeof(achDllName) - 5)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "name too long: %s\n", p1);
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(achDllName, p1, len);
|
|
|
|
strncpy(achDllName + len, ".dll", sizeof(achDllName) - len);
|
|
|
|
pszDllName = achDllName;
|
|
|
|
}
|
|
|
|
|
2015-02-04 23:56:23 +00:00
|
|
|
/* Open input file */
|
|
|
|
pszSourceFileName = argv[i];
|
|
|
|
file = fopen(pszSourceFileName, "r");
|
2011-05-16 13:12:07 +00:00
|
|
|
if (!file)
|
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
fprintf(stderr, "error: could not open file %s\n", pszSourceFileName);
|
2011-05-16 13:12:07 +00:00
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get file size */
|
|
|
|
fseek(file, 0, SEEK_END);
|
|
|
|
nFileSize = ftell(file);
|
|
|
|
rewind(file);
|
|
|
|
|
|
|
|
/* Allocate memory buffer */
|
|
|
|
pszSource = malloc(nFileSize + 1);
|
2011-09-19 07:29:36 +00:00
|
|
|
if (!pszSource)
|
|
|
|
{
|
|
|
|
fclose(file);
|
|
|
|
return -4;
|
|
|
|
}
|
2011-05-16 13:12:07 +00:00
|
|
|
|
|
|
|
/* Load input file into memory */
|
|
|
|
nFileSize = fread(pszSource, 1, nFileSize, file);
|
|
|
|
fclose(file);
|
|
|
|
|
|
|
|
/* Zero terminate the source */
|
|
|
|
pszSource[nFileSize] = '\0';
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
pexports = ParseFile(pszSource, file, &cExports);
|
|
|
|
if (pexports == NULL)
|
|
|
|
{
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
fprintf(stderr, "error: could not parse file!\n");
|
2019-08-18 13:32:22 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
[SPEC2DEF] Set ordinals explicitly in export def file
The reason is that dlltool orders the exports differently than MSVC builds (MSVC orders the exports by symbol name, rather than by export name), so we rely on sorting in the spec file, which was only respected, when ordinals were put into the def file.
On MSVC builds it is left to the linker to determine the correct order, which helps to get the differences between architectures right (different symbol decoration, difference between order for functions like NtLoadKey vs NtLoadKey2, which results from the stdcall decoration on x86, which is missing on other architectures.
TODO: To correctly handle non-x86 architectures with GCC builds, spec2def would need to reorder the export list based on symbol names, which would work for C functions, by taking the calling convention into account, but would require an extra c++-stdcall calling convention to be added to know the corresponding symbol starts with "?".
2019-09-22 12:59:09 +00:00
|
|
|
if (!gbMSComp)
|
|
|
|
{
|
|
|
|
if (ApplyOrdinals(pexports, cExports) < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "error: could not apply ordinals!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
if (pszDefFileName)
|
|
|
|
{
|
|
|
|
/* Open output file */
|
|
|
|
file = fopen(pszDefFileName, "w");
|
|
|
|
if (!file)
|
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
fprintf(stderr, "error: could not open output file %s\n", argv[i + 1]);
|
2011-05-16 13:12:07 +00:00
|
|
|
return -5;
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputHeader_def(file, pszDllName);
|
2019-08-18 13:32:22 +00:00
|
|
|
|
|
|
|
for (i = 0; i < cExports; i++)
|
|
|
|
{
|
|
|
|
if (pexports[i].bVersionIncluded)
|
|
|
|
OutputLine_def(file, &pexports[i]);
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pszStubFileName)
|
|
|
|
{
|
|
|
|
/* Open output file */
|
|
|
|
file = fopen(pszStubFileName, "w");
|
|
|
|
if (!file)
|
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
fprintf(stderr, "error: could not open output file %s\n", argv[i + 1]);
|
2011-05-16 13:12:07 +00:00
|
|
|
return -5;
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputHeader_stub(file);
|
2019-08-18 13:32:22 +00:00
|
|
|
|
|
|
|
for (i = 0; i < cExports; i++)
|
|
|
|
{
|
|
|
|
if (pexports[i].bVersionIncluded)
|
|
|
|
OutputLine_stub(file, &pexports[i]);
|
|
|
|
}
|
|
|
|
|
2011-05-16 13:12:07 +00:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pszLibStubName)
|
|
|
|
{
|
|
|
|
/* Open output file */
|
|
|
|
file = fopen(pszLibStubName, "w");
|
|
|
|
if (!file)
|
|
|
|
{
|
2015-02-04 23:56:23 +00:00
|
|
|
fprintf(stderr, "error: could not open output file %s\n", argv[i + 1]);
|
2011-05-16 13:12:07 +00:00
|
|
|
return -5;
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputHeader_asmstub(file, pszDllName);
|
2019-08-18 13:32:22 +00:00
|
|
|
|
|
|
|
for (i = 0; i < cExports; i++)
|
|
|
|
{
|
|
|
|
if (pexports[i].bVersionIncluded)
|
|
|
|
OutputLine_asmstub(file, &pexports[i]);
|
|
|
|
}
|
|
|
|
|
2014-09-05 20:07:53 +00:00
|
|
|
fprintf(file, "\n END\n");
|
2011-05-16 13:12:07 +00:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
2019-08-18 13:32:22 +00:00
|
|
|
free(pexports);
|
|
|
|
|
|
|
|
return 0;
|
2011-05-16 13:12:07 +00:00
|
|
|
}
|