[comp.lang.c++] C/C++ link in TC++

reed@m.cs.uiuc.edu (09/11/90)

I've been trying to link C object modules with C++, both created
using Turbo C++.  I must be doing something stupid, because the
Turbo linker is unable to find the C reference.  I've attached a
code sample below that fails when I do the following:

	tcc -c y.c
	tcc -c x.cpp y.obj

This fails to resolve the reference to "try"

Any ideas on what I'm overlooking?

Thanks in advance,

Dan Reed			reed@cs.uiuc.edu
Associated Professor
Department of Computer Science
University of Illinois
================================ y.c =========================
#include <stdio.h>

void try (int x)
{
	printf ("Value is %d\n", x);
}
================================ x.cpp =======================
#include <iostream.h>

extern void try(int x);

main()
{
	int	y;

	cout <<"Before call\n";

	y = 10;
	try (y);

	cout<<"After call\n";
}
========= end of x.cpp ==========

anto@vaxb.acs.unt.edu (09/11/90)

In article <4800101@m.cs.uiuc.edu>, reed@m.cs.uiuc.edu writes:
> I've been trying to link C object modules with C++, both created
[... Stuff deleted ...]
> ================================ x.cpp =======================
> #include <iostream.h>
> 
> extern void try(int x);

Try:

 extern "C" void try(int x);


Hope this helps.

Anto.

// no signature yet 8)

poffen@sj.ate.slb.com (Russ Poffenberger) (09/11/90)

In article <4800101@m.cs.uiuc.edu> reed@m.cs.uiuc.edu writes:
>
>
>I've been trying to link C object modules with C++, both created
>using Turbo C++.  I must be doing something stupid, because the
>Turbo linker is unable to find the C reference.  I've attached a
>code sample below that fails when I do the following:
>
>	tcc -c y.c
>	tcc -c x.cpp y.obj
>
>This fails to resolve the reference to "try"
>
>Any ideas on what I'm overlooking?
>
>Thanks in advance,
>
>Dan Reed			reed@cs.uiuc.edu
>Associated Professor
>Department of Computer Science
>University of Illinois
>================================ y.c =========================
>#include <stdio.h>
>
>void try (int x)
>{
>	printf ("Value is %d\n", x);
>}
>================================ x.cpp =======================
>#include <iostream.h>
>
>extern void try(int x);
>
>main()
>{
>	int	y;
>
>	cout <<"Before call\n";
>
>	y = 10;
>	try (y);
>
>	cout<<"After call\n";
>}
>========= end of x.cpp ==========


Try this in your c++ program...

extern "C" {
  void try(int);
}

as your function prototype in x.cpp.

The problem is with naming conventions. C++ uses new internal naming
conventions so it can distinguish between instances of overloaded functions
and such. Using the above syntax forces the compiler to use standard C naming
conventions.

Russ Poffenberger               DOMAIN: poffen@sj.ate.slb.com
Schlumberger Technologies       UUCP:   {uunet,decwrl,amdahl}!sjsca4!poffen
1601 Technology Drive		CIS:	72401,276
San Jose, Ca. 95110             (408)437-5254

cw@dptechno.UUCP (Charles Haden) (09/13/90)

In article <4800101@m.cs.uiuc.edu> reed@m.cs.uiuc.edu writes:
>
>
>I've been trying to link C object modules with C++, both created
>using Turbo C++.  I must be doing something stupid, because the
>Turbo linker is unable to find the C reference.  I've attached a
>code sample below that fails when I do the following:
>
>	tcc -c y.c
>	tcc -c x.cpp y.obj
>
>This fails to resolve the reference to "try"
>
>Any ideas on what I'm overlooking?

Your main problem lies primarily in the command line invocations
of the compiler. What I mean by that is your problem is not in
your source code it is in the way you are using the compiler.
As you should know from the manuals that you can compile both
"C" and "C++" source code with the compiler. What you probably
didn't read it that the compiler treats prototyping differently
based on the file extension ( default setting ). When compilling
a C++ source file, the function name is "tweaked" you might say
internally to allow for function overloading. This tweaking is
not normally done to standard C source code prototypes, since
overloading of function names is not allowed. There is a switch
however, that will force standard C source files to be tweaked
when compiling. ( see below ) A note of caution : this switch
can only be used when you have the source code and re-compile it.


Proper command line invocations ( tested with omitted source. ) :

	tcc -c -P y.c
	tcc x.cpp y.obj

grimlok@hubcap.clemson.edu (Mike Percy) (09/13/90)

cw@dptechno.UUCP (Charles Haden) writes:

>Your main problem lies primarily in the command line invocations
>of the compiler. What I mean by that is your problem is not in
>your source code it is in the way you are using the compiler.
>As you should know from the manuals that you can compile both
>"C" and "C++" source code with the compiler. What you probably
>didn't read it that the compiler treats prototyping differently
>based on the file extension ( default setting ). When compilling
>a C++ source file, the function name is "tweaked" you might say
>internally to allow for function overloading. This tweaking is
>not normally done to standard C source code prototypes, since
>overloading of function names is not allowed. There is a switch
>however, that will force standard C source files to be tweaked
>when compiling. ( see below ) A note of caution : this switch
>can only be used when you have the source code and re-compile it.
>Proper command line invocations ( tested with omitted source. ) :
>	tcc -c -P y.c
>	tcc x.cpp y.obj
 
EEEnnnnnttttt.  Close.  The -P option is instructing the compiler that
no matter what the extension of the file, assume the contents of the
file is C++ source.  This would force code.pas to be treated (probably
incorrectly) as C++ code.  This allows you to not follow Borland's .cpp
extension and use something else, perhaps Gnu's extension of .cc, or if
using a different command.com which allows '+' in filenames (DOS has no
problems with it, just the command.com parser) to use a more clear .c++
extension.  
 
Let's assume that the contents of y.c is really C code and not C++, or
better still that we have y.obj and y.h (for the prototypes) but not
y.c, but we know that y.obj came from a C compiler.  The change to make
to y.h is
 
#ifdef __cplusplus
extern "C" {
#endif
int try(int x);
#ifdef __cplusplus
}
#endif
 
This lets us use y.h in both the C compiler and the C++ compiler (hey
what a concept).
 
It always helps to look at the standard header files in a case like this
to see how it's done... 

"I don't know about your brain, but mine is really...bossy."
Mike Percy                    grimlok@hubcap.clemson.edu
ISD, Clemson University       mspercy@clemson.BITNET
(803)656-3780                 mspercy@clemson.clemson.edu

steve@taumet.com (Stephen Clamage) (09/13/90)

cw@dptechno.UUCP (Charles Haden) writes:

>In article <4800101@m.cs.uiuc.edu> reed@m.cs.uiuc.edu writes:
   ...
>>I've attached a code sample below that fails when I do the following:
>>	tcc -c y.c
>>	tcc -c x.cpp y.obj

>Your main problem lies primarily in the command line invocations
>of the compiler. What I mean by that is your problem is not in
>your source code it is in the way you are using the compiler.
>As you should know from the manuals that you can compile both
>"C" and "C++" source code with the compiler. What you probably
>didn't read it that the compiler treats prototyping differently
>based on the file extension ( default setting ). When compilling
>a C++ source file, the function name is "tweaked" you might say
>internally to allow for function overloading. This tweaking is
>not normally done to standard C source code prototypes, since
>overloading of function names is not allowed. There is a switch
>however, that will force standard C source files to be tweaked
>when compiling.

This explanation is not really correct.

Turbo C++ is really two compilers in one:  it is a C compiler, and it is
also a C++ compiler.  In the absence of compiler flags, a file with an
extension of ".CPP" will be compiled by the C++ compiler, and one with
an extension of ".C" will be compiled by the C compiler.  The "-P" flag
forces ".C" files to be compiled by the C++ compiler.  (I don't mean
that there are necessarily two complete compilers hiding in one .EXE
file, but that the behavior is as if there were two compilers.)

C and C++ are different languages, and have some different rules.  One
C++ rule is that any function not declared
	extern "C" ...
is expected to be a C++ function, and follows all C++ naming rules and
calling conventions.  If a function is declared so as to make it appear
to be a C++ function, but is really a C function, the program can be
expected to fail in some way.  The most common way to fail (true for
all current C++ systems I know of) is that the program will not link.
The link failure is due to the different ways function names are
treated in C and C++.  The discussion in this paragraph is not
unique to Turbo C++, but applies to any C++ implementation.
-- 

Steve Clamage, TauMetric Corp, steve@taumet.com