Forum: Compiler & IDEs Linkerproblem - C Code in C++ Code linken


von C. M. (chrisliebaer)


Lesenswert?

Hallo zusammen,

ich habe folgendes Problem:
Ich habe noch nicht viel mit dem GCC auf der AVR Plattform gearbeitet 
und bin mir daher nicht sicher ob das Problem an mir liegt, oder ob ich 
da villeicht gerade in einer noch nicht implementierten Sparte vom GCC 
rumstochere.

Es gibt eine main.cpp mit folgendem Inhalt
1
#include <io/uart.h>
2
3
int main()
4
{
5
  debugInit(9600);
6
7
}
Die Datei wird also mit g++ kompiliert. Das ist auch notwendig, also 
bitte keine Hinweise, dass ich einfach mit dem gcc kompilieren soll, ich 
brauch später C++.

Nun bau ich mir gerade eine Library für einige Funktionen, wie z.B. 
Initialisierung fürs UART; linke also diese dazu

Dort gibts es dann folgende Datei io/uart.c:
1
//[..]
2
void debugInit(unsigned pBaudRate)
3
{
4
  setDebugBaudRate(pBaudRate);
5
  
6
  //enable transmitter/receiver register
7
  UCSR0B |= 1 << TXEN0;
8
  UCSR0B |= 1 << RXEN0;
9
  
10
  //set frame format: 8data, 1 stop bits
11
  UCSR0C = (1 << USBS0) | (3 << UCSZ00);
12
  
13
  stdout = &mystdout;
14
}
15
//[..]

Und dazu natürlich noch die Header Datei:
1
#pragma once
2
3
#include <inttypes.h>
4
#include <stdio.h>
5
6
void debugInit(unsigned pBaudRate);
7
//[..]
Dies wird mit dem GCC kompiliert, also kein C++ an dieser Stelle.

Das problem ist nun, dass der Linker bei obigen Code motz, dass er 
debugInit(unsigned pBaudRate); nicht linken konnte.

Kompiliere ich die main.cpp als C Code geht es.

Hat da jemand eine erklärung für? Ich finde einfach meinen Fehler nicht 
und bin auch nicht sicher ob es nicht villeicht sogar ein Problem mit 
dem Linker an der stelle gibt, wenn C mit C++ geklinkt wird. Wobei das 
ja eigentlich egal sein müsste, da der Code zu diesem Zeitpunkt ja 
bereits kompiliert ist.

Vielen Dank schonmal

Gruß
Chris

von Klaus W. (mfgkw)


Lesenswert?

1
#include <inttypes.h>
2
#include <stdio.h>
3
4
#ifdef  __cplusplus
5
extern "C" {
6
#endif
7
8
void debugInit(unsigned pBaudRate);
9
//[..]
10
11
#ifdef  __cplusplus
12
}
13
#endif

In der io/uart.c muß natürlich auch die .h #includet werden.

von Hmmmm (Gast)


Lesenswert?

Chris R. schrieb:
> Das problem ist nun, dass der Linker bei obigen Code motz, ...

Das ist keine Fehlermeldung. Die selbige wäre aber interessant.

von C. M. (chrisliebaer)


Lesenswert?

Erstmal vielen Dank für die Idee mit dem eingebetteten C Code, auf die 
Idee bin ich garnicht gekommen, aber das ist ja auch keine Erklärung für 
das Problem, sondern nur ein Workaround.

Fehlermeldung vom Kompiler:
1
./main.o: In function `main':
2
main.cpp:(.text+0x4): undefined reference to `debugInit(unsigned int)'

von Hmmmm (Gast)


Lesenswert?

Klaus Wachtler schrieb:
> #ifdef  __cplusplus
> extern "C" {
> #endif

Ist das nicht die Lösung, wenn man 'C' Code aus 'C++' heraus aufrufen 
will?
Der TO möchte genau das andere, aus 'C++ ' eine 'C' Funktion aufrufen.
Aber vielleicht hilft es in beiden Fällen, weiß ich gerade nicht.

von Klaus W. (mfgkw)


Lesenswert?

PS: include-guards statt #pragma... wäre portabler:
1
#ifdef _IO_UART_H_
2
#define _IO_UART_H_
3
4
#include <inttypes.h>
5
#include <stdio.h>
6
7
#ifdef  __cplusplus
8
extern "C" {
9
#endif
10
11
void debugInit(unsigned pBaudRate);
12
//[..]
13
14
#ifdef  __cplusplus
15
}
16
#endif
17
18
#endif /* ifndef _IO_UART_H_ */

von Klaus W. (mfgkw)


Lesenswert?

Hmmmm schrieb:
> Klaus Wachtler schrieb:
>> #ifdef  __cplusplus
>> extern "C" {
>> #endif
>
> Ist das nicht die Lösung, wenn man 'C' Code aus 'C++' heraus aufrufen
> will?
> Der TO möchte genau das andere, aus 'C++ ' eine 'C' Funktion aufrufen.
> Aber vielleicht hilft es in beiden Fällen, weiß ich gerade nicht.

Ich nehme an, daß seine (mir unbekannten Fehler) davon kommen, daß bei 
C++ per default name mangling stattfindet.
Dann wird in seiner CPP-Datei also aus dem Funktionsnamen n der .h etwas 
anderes gemacht, als wenn er die andere Datei mit C kompiliert.

dagegen hilft, die C-Funktionen als extern "C" zu deklarieren.
Nötig ist es also hier in jedem Fall; ob es alle seine Probleme löst muß 
man abwarten.

von Klaus W. (mfgkw)


Lesenswert?

Chris R. schrieb:
> aber das ist ja auch keine Erklärung für
> das Problem, sondern nur ein Workaround.

Dieser Workaround ist gängige Praxis.

von Rolf M. (rmagnus)


Lesenswert?

Klaus Wachtler schrieb:
> Chris R. schrieb:
>> aber das ist ja auch keine Erklärung für das Problem, sondern nur ein
>> Workaround.
>
> Dieser Workaround ist gängige Praxis.

Außerdem ist dieser Workaround kein Workaround, sondern der dafür 
vorgesehene Weg.

Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.