I moved the discussion under a more informative heading.On Saturday, December 13, 2014 12:12:13 AM UTC-5, gmhwxi wrote:
By Barry Schwartz:
On Saturday, December 13, 2014 12:10:17 AM UTC-5, gmhwxi wrote:
john skaller ska...@users.sourceforge.net skribis:
But its WORSE. In both C and C++ you ALSO have to say something like:
#if !defined(FLX_STATIC_LINK) && FLX_WIN32
#define FLX_EXPORT __declspec(dllexport)
#define FLX_IMPORT __declspec(dllimport)
#else
#define FLX_EXPORT
#define FLX_IMPORT
#endif#ifdef BUILD_RTL
#define RTL_EXTERN FLX_EXPORT
#else
#define RTL_EXTERN FLX_IMPORT
#endifetc.
I asked about this a few weeks ago. Yes, it can be done. It is not
done by using =“ext:” but by a more convoluted but otherwise better
method; I am not the one to discuss it, however.On Saturday, December 13, 2014 12:08:42 AM UTC-5, gmhwxi wrote:
By John Skaller:
On 13/12/2014, at 2:59 PM, Brandon Barker wrote:
In the wiki article there is this, which I think is what you are
referring to:Make an ATS function available in C (can be implemented in ATS or C):
extern
fun myfun (n: int, res: int): int = “ext#myfun_in_c” (* call as
myfun_in_c(int, int) in C *)(* Have the C function name be the same as the ATS function name: )
extern
fun myfun (n: int, res: int): int = “ext#” ( call as myfun(int, int)
in C *)That’s not enough in C++. You have to say
// in header file:
extern “C” void name (int);// in body code:
void name ( int x) { … }But its WORSE. In both C and C++ you ALSO have to say something like:
#if !defined(FLX_STATIC_LINK) && FLX_WIN32
#define FLX_EXPORT __declspec(dllexport)
#define FLX_IMPORT __declspec(dllimport)
#else
#define FLX_EXPORT
#define FLX_IMPORT
#endif#ifdef BUILD_RTL
#define RTL_EXTERN FLX_EXPORT
#else
#define RTL_EXTERN FLX_IMPORT
#endifHere “BUILD_RTL” has to be supplied as a C/C++ compiler switch when
building
the DLL and not otherwise. FLX_STATIC_LINK has to be supplied if
compiling
for static linkage (and not otherwise).Now the functions have to be declared like
RTL_EXTERN void flx_trace(flx_trace_t* tr,flx_range_srcref_t sr, char
const *file, int line, char const *msg);in C++ for a C++ function and
extern “C”
as well if you want to use C liker names. This is usually done like:
#ifdef __cplusplus /* support use by C++ code */
extern “C” {
#endifRTL_EXTERN void flx_trace(flx_trace_t* tr,flx_range_srcref_t sr, char
const *file, int line, char const *msg);#ifdef __cplusplus
}
#endifIt used to be only Windows required dllimport/dllexport but this is NOT
so any more.
Now GNU linkers (finally) have visibility control, you have to do it
with all modern gcc
systems (i.e. all Unix, OSX and Linux) platforms too (unless you want to
export everything!)So clearly this is all non-trivial and it HAS to be supported by ATS
compiler.
I’m pretty sure ATS compiler cannot do the DLL import/export at present.
In particular(a) the ATS to C compiler has to generate the appropriate code AND
(b) the C/C++ compiler/linker toolchain invoked by ATS driver
program has to provide the required STATIC_LINK and BUILD_LIB switches
as well.I don’t care about (b), I can organise C/C++ compile/link myself.
All this is in addition to ensuring the C code works under C++ compiler
as well.
(no implicit casts from void * allowed, no assuming enums are ints, no
funny business
with bool either … )Even if you compile the library with C, the header files MUST work in
C++
as well as C.getting all this right is not hard but the method is NOT obvious: more
than half the
open source projects in existence get it wrong.The macro method for building DLLs above is the ONLY correct way to do
it.In turn this means when you’re translating a library from ATS to C you
MUST
supply the build macro name to the ATS translator. Which I don’t think
ATS
supports at present either.So … I may be wrong … but I do not think ATS can generate libraries
yet.Just as an aside, although messy this is the correct protocol:
export sym; // in provider import sym; // in consumer
C, as usual, gets everything wrong and tries to just use
extern sym;
for both purposes and that simply does not work. [In particular
the statementextern int x;
has unspecified meaning: you can’t tell if it’s a declaration or
definition,
so C’s hackery doesn’t even work in C]