In C its possible to define macros. Macros are not available in Delphi, so functions must be used to translate C-macros. In most cases its easier to translate a macro based on the information about it from the documentation than trying to translate the code directly. One example of a macro: Here, quite clearly, the macro accepts integers. A valid translation of this macro is The following macro accepts any datatype as parameter: It compares the values of parameter a with parameter b and returns the higher value. So that our function can pass any datatype, we can use variants. I do recommend implementing the Delphi translation on the basis of the macros documentation. Sometimes it is not possible to translate macros so that there is a 1:1 correspondence between the C code and the Delphi translation. For example, some header files use macros in #define statements for declaring constants. Here is an example from the clusapi header file. The ensuing declarations use a macro named CLUSTER_PROPERTY_SYNTAX to calculate the values of the constants: We cannot do it like this in Delphi and, besides, Delphi does not allow the value of an enumeration entry to be assigned. The macro must be resolved in order to do the translation to Delphi. We can translate the #define statement into the following Delphi function: However, function calls are not possible in constant declarations. To resolve that, we declare the constants as follows, without wrapping them together as the C macro?s enumeration has done: Actually, this is a case where I would recommend translating the macro as a whole into a function, as it might be useful to make it available to applications.
#define PRIMARYLANGID(lgid) ((WORD )(lgid) & 0x03ff)
Function PRIMARYLANGID (lgid: DWord): DWord;Begin Result := lgid and $03FFEnd;
#define max(a,b) (((a) > (b)) ? (a) : (b))
Function Max (A, B: Variant): Variant;Begin If A > B then Result := A else Result := BEnd;
#define CLUSPROP_SYNTAX_VALUE( type, format ) ((DWORD) ((type << 16) | format))
typedef enum CLUSTER_PROPERTY_SYNTAX { CLUSPROP_SYNTAX_ENDMARK = CLUSPROP_SYNTAX_VALUE( CLUSPROP_TYPE_ENDMARK, CLUSPROP_FORMAT_UNKNOWN ), CLUSPROP_SYNTAX_NAME = CLUSPROP_SYNTAX_VALUE( CLUSPROP_TYPE_NAME, CLUSPROP_FORMAT_SZ ), CLUSPROP_SYNTAX_RESCLASS = CLUSPROP_SYNTAX_VALUE( CLUSPROP_TYPE_RESCLASS, CLUSPROP_FORMAT_DWORD ),
Function CLUSPROP_SYNTAX_VALUE (dwType: DWORD; dwFormat: DWORD): DWORD;Begin Result := (dwType shl 16) or dwFormat;End;
CONST CLUSPROP_SYNTAX_ENDMARK = CLUSPROP_TYPE_ENDMARK shl 16 or CLUSPROP_FORMAT_UNKNOWN; CLUSPROP_SYNTAX_NAME = CLUSPROP_TYPE_NAME shl 16 or CLUSPROP_FORMAT_SZ; CLUSPROP_SYNTAX_RESCLASS = CLUSPROP_TYPE_RESCLASS shl 16 or CLUSPROP_FORMAT_DWORD;
