` Printed Icetips Article

Icetips Article



Par2: C Programming in Clarion
1998-05-18 -- Sean Wilson
 
The Clarion development system has been shipping with a C compiler for some time and 
yet many people are still running into problems developing Multi-Language applications. 
 The Programmer's Guide contains a single chapter on Multi-Language Programming 
which is a must read, despite the fact that it was written about 5 years ago and some of 
the information is no longer accurate or relevant. 
 Multi-Language Programming deserves a detailed discussion, however this article is 
intended only to illustrate the most fundamental aspects of calling C code from a Clarion

application. 
 Those of you that use the Application Generator for developing every application will 
have to excuse the fact that I'm an unrepentant hand-coder. 

The Anatomy of a C++ Source File 

Those of you who are still conscious are probably saying "C++, why would I want to 
know about C++? I want to call C code!" The problem is that the 'C' compiler that 
ships with Clarion is really a C++ compiler. It does the same job, but there are some 
important differences, I won't go into all the subtle differences between C & C++, I will

simply discuss those that are likely to affect the way you have to code for compatibility
with Clarion. 
 The best way to examine a programming issue is to look at some code: 


 1: // TSTC.C
 2: #if defined(__cplusplus)
 3: extern "C" {
 4: #endif
 5: int add(int a, int b);
 6: #if defined(__cplusplus)
 7: };
 8: #endif
 9:
10: int add(int a, int b)
11: { return a+b;
12: }
This possibly looks a little more complicated than you first anticipated, I will now 
describe in detail what is going on at each line of code: 

1 C++ line comments begin with '//', the comment is simply to tell you what the C file 
in this example is called. 
2 This introduces a section of conditionally compiled code. The __cplusplus macro is 
defined whenever a C++ compiler is being used to compile the code. 
3 The extern "C" directive is used to declare a name that will use C language naming 
conventions. By default C++ compilers mangle or decorate names by encoding 
additional type information into the name to facilitate type-safe linking. This directive
will disable the name-mangling and prefix the name with an underscore ('_'). 
4 This terminates a section of conditionally compiled code. 
5 The prototype for the C function. As it appears within the extern "C" directive the 
linkage name generated byt the compiler will be "_add". 
7 This terminates the extern "C" directive. 
10-12 The definition of the C function. Note that C++ compilers require ANSI syntax
which expects the definition to be fully prototyped with the parameter types included 
in the parameter list. 


Calling C From Clarion 

The Clarion code is simpler to explain, particularly as you are probably quite familiar
with
the language, still taking a similar approach: 
 1:! TSTCLW.CLW  
 2:  PROGRAM
 3:
 4:  MAP
 5:    MODULE('TstC')
 6:      add(SIGNED a, SIGNED b), SIGNED, PROC, RAW, NAME('_add')
 7:    END
 8:  END
 9:
10:  CODE
11:  MESSAGE(add(2,2))
This code does very little, in fact nothing useful other than to demonstrate the success 
of calling the add() function by displaying its' result. 

5 The MODULE() statement is required to prototype externally defined procedures, 
however the name that appears is not used. 
6 This is where the hard work is done. Note that the prototype should match exactly 
the C++ prototype otherwise some very odd problems will occur. Parameter prototyping 
errors are usually very difficult to track down. The NAME() attribute should contain the 
linkage name for the function which as noted before is "_add". Although it is not 
required for this function, it is good practice to include the RAW attribute which
ensures
that additional type iformation is not passed as an additional implicit parameter. I
usually 
like to include the PROC attribute which suppresses warnings when functions are 
called as procedures.  


Putting it all Together 

The only thing missing is the project file, this couldn't be simpler and is easily created
in 
the Clarion IDE. I have included it here for completeness but it deserves no additional 
explanation: 
-- tstc
#noedit
#system win32
#model clarion dll
#pragma debug(vid=>full)
#compile tstclw.clw
#compile tstc.c
#link tstc.exe


And Finally... 

That's it! Nothing simpler really, however I realise I have only scratched the surface 
here. If there are specific issues that you would like to see addressed then let me know,
I will gladly incorporate new information into future articles.



Printed November 21, 2024, 12:33 pm
This article has been viewed/printed 35353 times.