gas@cs.nott.ac.uk (Alan Shepherd) (04/16/91)
From my understanding, the correct way to make C++ functions accessible from modules compiled with c is to delcare them extern "C" as in the following example: #include <stream.h> extern "C" void tryCout(); void tryCout() { cout << "Hello World\n"; } Now this can be called from a program compiled with cc. For example, the following program: extern void tryCout(); main() { tryCout(); } This happily compiles without any warnings. However, when running, a segv occurs in the cout operation. This segv disappears if both modules are compiled with CC instead of just the first. Adding another level of indirection to the code by writing another function to call tryCout doesn't work either. Is this a bug ? It certainly looks like it ! Alan Shepherd
gribble@ogre.cica.indiana.edu (04/18/91)
I think you have it backwards--use extern to link C functions to C++. It has to do w/ "name-mangling"--something C++ does to function names. -- ************************************************************************** * Steve Gribble (812) 855-9172/7629 gribble@cica.cica.indiana.edu * Systems Manager, Inst. of Social Research swg@socmail.soc.indiana.edu * Dept. of Sociology, Indiana University gribble@iubacs
jac@gandalf.llnl.gov (James A. Crotinger) (04/18/91)
I'm not certain, but I believe that in order to use streams, the main() program needs to be written in C++. The problem is that cout is a globally declared class which must be initialized before main() is called. This won't happen if the main() is written in C. For example, the following works fine on my Sun (using Sun C 1.0 and Sun C++ 2.0). ----------------------------------- // main.cc #include <stream.h> extern "C" void foo(void); extern "C" void bar(void); main() { bar(); } void foo() { cout << "Testing...\n"; } ----------------------------------- /* bar.c */ extern void foo(); void bar() { foo(); } ----------------------------------- Jim -- ----------------------------------------------------------------------------- James A. Crotinger Lawrence Livermore Natl Lab // The above views jac@moonshine.llnl.gov P.O. Box 808; L-630 \\ // are mine and are not (415) 422-0259 Livermore CA 94550 \\/ necessarily those of LLNL
haydens@natasha.juliet.ll.mit.edu (Hayden Schultz) (04/18/91)
I'm pretty sure that Jim (jac@gandalf.llnl.gov) has the right answer, that because the C++ main does tricky things like call global constructors that the original C main didn't call. So it makes sense that when you try to access a global object that hasn't been initialized correctly, like cout, you get a segv. But isn't there some way to explicitly invoke the global constructors and destructors so you can use a C main and call C++ routines? I can't manage to accomplish this with Sun CC. It would be very nice if I could use C++ routines everywhere I use C routines. For example, I can call C routines from lisp, but since it doesn't have a C++ main, I can't figure out how to call C++ routines. Hayden Schultz (haydens@juliet.ll.mit.edu) MIT Lincoln Lab
steve@taumet.com (Stephen Clamage) (04/18/91)
gas@cs.nott.ac.uk (Alan Shepherd) writes: >#include <stream.h> >extern "C" void tryCout(); >void tryCout() >{ > cout << "Hello World\n"; >} >Now this can be called from a program compiled with cc. For example, >the following program: >extern void tryCout(); >main() >{ > tryCout(); >} >This happily compiles without any warnings. However, when running, a >segv occurs in the cout operation. This segv disappears if both >modules are compiled with CC instead of just the first. Almost all current C++ compilers require that main() be compiled by the C++ compiler, even if other modules are compiled with a compatbile C compiler. Your compiler manual should tell you this. In general, C++ programs will require initialization of static objects via their constructors at runtime; cout is such an object. The C++ compiler arranges for the constructors (and destructors) to be called via special code emitted in the module in which main() appears. This ensures the special code is emitted and called exactly once. In your example, main() was not compiled by the C++ compiler, cout was not initialized, and the program failed when it tried to use member functions of cout. -- Steve Clamage, TauMetric Corp, steve@taumet.com
grp@Unify.com (Greg Pasquariello) (04/20/91)
In article <1991Apr16.122905.23613@cs.nott.ac.uk>, gas@cs.nott.ac.uk (Alan Shepherd) writes: > > From my understanding, the correct way to make C++ functions > accessible from modules compiled with c is to delcare them extern "C" > as in the following example: > > #include <stream.h> > > extern "C" void tryCout(); > > void tryCout() > { > cout << "Hello World\n"; > } > > Now this can be called from a program compiled with cc. For example, > the following program: > > extern void tryCout(); > > main() > { > tryCout(); > } > > This happily compiles without any warnings. However, when running, a > segv occurs in the cout operation. This segv disappears if both > modules are compiled with CC instead of just the first. Adding > another level of indirection to the code by writing another function > to call tryCout doesn't work either. > > Is this a bug ? It certainly looks like it ! You need to declare, in your C++ source code, an object of class Iostream_init. The constructor for the class will initialize cin, cout, cerr, and clog, and your core dump will go away. Note that this relies on C++ 2.0 and the iostreams package. To try it out, do: #include <iostream.h> extern "C" void tryCout(); void tryCout() { Iostream_init io; cout << "Hello World\n"; } On the C side: extern void tryCout(); main() { tryCout(); } > Alan Shepherd -- --- Greg Pasquariello grp@unify.com Unify Corporation Be good and never poison people
steve@taumet.com (Stephen Clamage) (04/21/91)
haydens@natasha.juliet.ll.mit.edu (Hayden Schultz) writes: >But isn't there some way to explicitly invoke the global constructors >and destructors so you can use a C main and call C++ routines? I can't >manage to accomplish this with Sun CC. There is no portable way to do this now. The ANSI C++ committee X3J16 is investigating ways to accomplish this. -- Steve Clamage, TauMetric Corp, steve@taumet.com
niklas@appli.se (Niklas Hallqvist) (04/21/91)
haydens@natasha.juliet.ll.mit.edu (Hayden Schultz) writes: >I'm pretty sure that Jim (jac@gandalf.llnl.gov) has the right answer, >that because the C++ main does tricky things like call global >constructors that the original C main didn't call. So it makes sense >that when you try to access a global object that hasn't been >initialized correctly, like cout, you get a segv. >But isn't there some way to explicitly invoke the global constructors >and destructors so you can use a C main and call C++ routines? I can't >manage to accomplish this with Sun CC. >It would be very nice if I could use C++ routines everywhere I use C >routines. For example, I can call C routines from lisp, but since it >doesn't have a C++ main, I can't figure out how to call C++ routines. Yes, that's tricky, but it ought to be possible with a little hacking. If your lisp implementation supports calling lisp functions from C/C++, and you have a binary editor, you should be able to patch in another name for the C++ "main", e.g "cppm" or something like that, and then call it from lisp the usual way. When this "main" gets called it should execute the global constructors for your C++ program, and then do a callback to your lisp function, which later will call other C++ functions. At the time of the return of this callback function the C++ "main" should execute the global destuctors before returning to lisp. This is of course very unportable since not every compiler puts the global constructor calling in "main", but rather before "main" is called, though I think AT&T derived C++ implementations will do it inside main. I also think that a binary editor won't be necessary in such implementations since they generate C output which could be edited with e.g. a sed script to change the name of the "main" function. Hope this helps Niklas -- Niklas Hallqvist Phone: +46-(0)31-40 75 00 Applitron Datasystem Fax: +46-(0)31-83 39 50 Molndalsvagen 95 Email: niklas@appli.se S-412 63 GOTEBORG, Sweden mcsun!sunic!chalmers!appli!niklas
grp@Unify.com (Greg Pasquariello) (04/22/91)
In article <675@taumet.com>, steve@taumet.com (Stephen Clamage) writes: > > Almost all current C++ compilers require that main() be compiled by > the C++ compiler, even if other modules are compiled with a compatbile > C compiler. Your compiler manual should tell you this. > > In general, C++ programs will require initialization of static objects > via their constructors at runtime; cout is such an object. The C++ > compiler arranges for the constructors (and destructors) to be called > via special code emitted in the module in which main() appears. This > ensures the special code is emitted and called exactly once. Only if some objects need initialization, and cannot provide it themselves! Iostreams with 2.0 provide an initializer object, called Iostream_init, so there is no reason you cannot use it unless you don't have the iostream library or a similar method of initializing your objects. It's true, I read it in the manual! :-) > > In your example, main() was not compiled by the C++ compiler, cout > was not initialized, and the program failed when it tried to use > member functions of cout. > -- > > Steve Clamage, TauMetric Corp, steve@taumet.com -- --- Greg Pasquariello grp@unify.com Unify Corporation Be good and never poison people
tell@oscar.cs.unc.edu (Stephen Tell) (04/24/91)
In article <1991Apr22.081254@Unify.com> grp@Unify.com (Greg Pasquariello) writes: >In article <675@taumet.com>, steve@taumet.com (Stephen Clamage) writes: >> >> Almost all current C++ compilers require that main() be compiled by >> the C++ compiler, even if other modules are compiled with a compatbile >> C compiler. Your compiler manual should tell you this. >> >> In general, C++ programs will require initialization of static objects >> via their constructors at runtime; cout is such an object. The C++ >> compiler arranges for the constructors (and destructors) to be called >> via special code emitted in the module in which main() appears. This >> ensures the special code is emitted and called exactly once. >[Iostreams with 2.0 provide an initializer object...] >[but not all class libraries do] >> In your example, main() was not compiled by the C++ compiler, cout >> was not initialized, and the program failed when it tried to use >> member functions of cout. Another C++ hacker and I figured out a way to do this for a simulator that was written in C and allows user code to be linked in with the program. Patch the .o files from the system so that the C main is renamed to somthing like "MaIn." Write short C++ file consisting of: extern "C" MaIn(int argc; char **argv); main(int argc, char **argv) { return MaIn(argc, argv); } This way, the real main is written in C++ and the constructors get called. The other code that you link in with the system can then be C++ and use whatever classes it likes since static objects will have had their constructors called. This may not be ideal, but it works. We did the C++ stuff with g++. I don't know what C compiler the original program was written with. It requires that the product be shipped with ordinary .o files that you run "ld" on to make customized versions, instead of funky schemes to attach user code to a running program. >> Steve Clamage, TauMetric Corp, steve@taumet.com >Greg Pasquariello grp@unify.com >Unify Corporation Be good and never poison people Steve -- Steve Tell tell@cs.unc.edu H: +1 919 968 1792 #5L Estes Park apts CS Grad Student, UNC Chapel Hill. W: +1 919 962 1845 Carrboro NC 27510 Duke Blue Devils: 1991 NCAA Basketball National Champions! We're Number 1 !! UNLV 90-91 record: "34 and DUKE."
bright@nazgul.UUCP (Walter Bright) (04/27/91)
In article <675@taumet.com>, steve@taumet.com (Stephen Clamage) writes:
/ Almost all current C++ compilers require that main() be compiled by
/ the C++ compiler, even if other modules are compiled with a compatbile
/ C compiler.
/
/ In general, C++ programs will require initialization of static objects
/ via their constructors at runtime; cout is such an object. The C++
/ compiler arranges for the constructors (and destructors) to be called
/ via special code emitted in the module in which main() appears.
This is not the way Zortech C/C++ works. There, the static constructors
are called by the startup code before main() is called, so it doesn't
matter if main() was compiled with the C or the C++ compiler. In fact
the runtime library for the C compiler uses static constructors... (!).
mike@taumet.com (Michael S. Ball) (04/29/91)
In article <306@nazgul.UUCP> bright@nazgul.UUCP (Walter Bright) writes: >In article <675@taumet.com>, steve@taumet.com (Stephen Clamage) writes: >/ In general, C++ programs will require initialization of static objects >/ via their constructors at runtime; cout is such an object. The C++ >/ compiler arranges for the constructors (and destructors) to be called >/ via special code emitted in the module in which main() appears. > >This is not the way Zortech C/C++ works. There, the static constructors >are called by the startup code before main() is called, so it doesn't >matter if main() was compiled with the C or the C++ compiler. In fact In fact, C++ compilers which define their own linker format, or which have access to a linker format which provides multiple named sections (psects, etc) can and should do it this way. The Borland C++ system does exactly that, as does the Oregon Software VMS compiler. Unfortunately, the majority of the C++ implementations for other environments are forced to cope with the antequated UNIX linkers which don't give you any way to do this. It appears that ELF will finally let UNIX implementors handle initialization rationally. The consequence is that the majority of different C++ compilers require that the main program be compiled with C++, while those compilers with the largest number of instances each (MSDOS compilers) don't. On a UNIX-oriented network, Steve's comments are accurate. -- Michael S. Ball mike@taumet.com TauMetric Corporation (619)697-7607