diff options
Diffstat (limited to 'python/sip/sipgen/gencode.c')
-rw-r--r-- | python/sip/sipgen/gencode.c | 10820 |
1 files changed, 10820 insertions, 0 deletions
diff --git a/python/sip/sipgen/gencode.c b/python/sip/sipgen/gencode.c new file mode 100644 index 00000000..b23ee2b1 --- /dev/null +++ b/python/sip/sipgen/gencode.c @@ -0,0 +1,10820 @@ +/* + * The code generator module for SIP. + * + * Copyright (c) 2007 + * Riverbank Computing Limited <[email protected]> + * + * This file is part of SIP. + * + * This copy of SIP is licensed for use under the terms of the SIP License + * Agreement. See the file LICENSE for more details. + * + * SIP is supplied WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + */ + + +#include <stdio.h> +#include <time.h> +#include <errno.h> +#include <stdlib.h> +#include <stdarg.h> +#include <string.h> + +#include "sip.h" + + +/* + * These must match the values of SIP_TYPE_FLAGS_SHIFT and SIP_TYPE_FLAGS_MASK + * in siplib/sip.h. + */ +#define TYPE_FLAGS_SHIFT 8 +#define TYPE_FLAGS_MASK 0x0f00 + + +/* Control what generateSingleArg() actually generates. */ +typedef enum { + Call, + Declaration, + Definition +} funcArgType; + + +/* An entry in the sorted array of methods. */ +typedef struct { + memberDef *md; /* The method. */ + int is_static; /* Set if all overloads are static. */ +} sortedMethTab; + + +static int currentLineNr; /* Current output line number. */ +static char *currentFileName; /* Current output file name. */ +static int previousLineNr; /* Previous output line number. */ +static char *previousFileName; /* Previous output file name. */ +static int exceptions; /* Set if exceptions are enabled. */ +static int tracing; /* Set if tracing is enabled. */ +static int generating_c; /* Set if generating C. */ +static int release_gil; /* Set if always releasing the GIL. */ +static const char *prcode_last = NULL; /* The last prcode format string. */ +static int prcode_xml = FALSE; /* Set if prcode is XML aware. */ + + +static void generateDocumentation(sipSpec *, char *); +static void generateBuildFile(sipSpec *, char *, char *, int); +static void generateInternalAPIHeader(sipSpec *, char *, stringList *); +static void generateCpp(sipSpec *, char *, char *, int *); +static void generateIfaceCpp(sipSpec *, ifaceFileDef *, char *, char *, + FILE *); +static void generateMappedTypeCpp(mappedTypeDef *, FILE *); +static void generateImportedMappedTypeHeader(mappedTypeDef *mtd, sipSpec *pt, + FILE *fp); +static void generateMappedTypeHeader(mappedTypeDef *, int, FILE *); +static void generateClassCpp(classDef *cd, sipSpec *pt, FILE *fp); +static void generateImportedClassHeader(classDef *cd, sipSpec *pt, FILE *fp); +static void generateClassTableEntries(sipSpec *pt, nodeDef *nd, FILE *fp); +static void generateClassHeader(classDef *, int, sipSpec *, FILE *); +static void generateClassFunctions(sipSpec *, classDef *, FILE *); +static void generateShadowCode(sipSpec *, classDef *, FILE *); +static void generateFunction(sipSpec *, memberDef *, overDef *, classDef *, + classDef *, FILE *); +static void generateFunctionBody(sipSpec *, overDef *, classDef *, classDef *, + int deref, FILE *); +static void generateTypeDefinition(sipSpec *pt, classDef *cd, FILE *fp); +static void generateTypeInit(sipSpec *, classDef *, FILE *); +static void generateCppCodeBlock(codeBlock *, FILE *); +static void generateUsedIncludes(ifaceFileList *, int, FILE *); +static void generateIfaceHeader(sipSpec *, ifaceFileDef *, char *); +static void generateShadowClassDeclaration(sipSpec *, classDef *, FILE *); +static int hasConvertToCode(argDef *ad); +static void deleteTemps(signatureDef *sd, FILE *fp); +static void gc_ellipsis(signatureDef *sd, FILE *fp); +static void generateArgs(signatureDef *, funcArgType, FILE *); +static void generateVariable(argDef *, int, FILE *); +static void generateNamedValueType(argDef *, char *, FILE *); +static void generateSingleArg(argDef *, int, funcArgType, FILE *); +static void generateBaseType(argDef *, FILE *); +static void generateNamedBaseType(argDef *, char *, FILE *); +static void generateTupleBuilder(signatureDef *, FILE *); +static void generateEmitters(sipSpec *pt, classDef *cd, FILE *fp); +static void generateEmitter(sipSpec *, classDef *, visibleList *, FILE *); +static void generateVirtualHandler(sipSpec *, virtHandlerDef *, FILE *); +static void generateVirtHandlerErrorReturn(argDef *res, FILE *fp); +static void generateVirtualCatcher(sipSpec *, classDef *, int, virtOverDef *, + FILE *); +static void generateUnambiguousClass(classDef *cd, classDef *scope, FILE *fp); +static void generateProtectedEnums(sipSpec *, classDef *, FILE *); +static void generateProtectedDeclarations(classDef *, FILE *); +static void generateProtectedDefinitions(classDef *, FILE *); +static void generateProtectedCallArgs(overDef *od, FILE *fp); +static void generateConstructorCall(classDef *, ctorDef *, int, FILE *); +static void generateHandleResult(overDef *, int, char *, FILE *); +static void generateOrdinaryFunction(sipSpec *, classDef *, memberDef *, + FILE *); +static void generateSimpleFunctionCall(fcallDef *, FILE *); +static void generateFunctionCall(classDef *cd, classDef *ocd, overDef *od, + int deref, FILE *fp); +static void generateCppFunctionCall(classDef *cd, classDef *ocd, overDef *od, + FILE *fp); +static void generateSlotArg(signatureDef *sd, int argnr, FILE *fp); +static void generateBinarySlotCall(overDef *od, char *op, int deref, FILE *fp); +static void generateNumberSlotCall(overDef *od, char *op, FILE *fp); +static void generateVariableHandler(varDef *, FILE *); +static int generateObjToCppConversion(argDef *, FILE *); +static void generateVarClassConversion(varDef *, FILE *); +static void generateVarMember(varDef *vd, FILE *fp); +static int generateVoidPointers(sipSpec *, classDef *, FILE *); +static int generateChars(sipSpec *, classDef *, FILE *); +static int generateStrings(sipSpec *, classDef *, FILE *); +static sortedMethTab *createFunctionTable(classDef *, int *); +static sortedMethTab *createMethodTable(classDef *, int *); +static int generateMethodTable(classDef *, FILE *); +static void generateEnumMacros(sipSpec *pt, classDef *cd, FILE *fp); +static int generateEnumMemberTable(sipSpec *, classDef *, FILE *); +static int generateInts(sipSpec *, classDef *, FILE *); +static int generateLongs(sipSpec *, classDef *, FILE *); +static int generateUnsignedLongs(sipSpec *, classDef *, FILE *); +static int generateLongLongs(sipSpec *, classDef *, FILE *); +static int generateUnsignedLongLongs(sipSpec *, classDef *, FILE *); +static int generateVariableType(sipSpec *pt, classDef *cd, argType atype, + const char *eng, const char *s1, const char *s2, FILE *fp); +static int generateDoubles(sipSpec *, classDef *, FILE *); +static int generateEnums(sipSpec *, classDef *, FILE *); +static int generateClasses(sipSpec *, classDef *, FILE *); +static void generateEnumsInline(sipSpec *, FILE *); +static void generateClassesInline(sipSpec *, FILE *); +static void generateAccessFunctions(sipSpec *, classDef *, FILE *); +static void generateConvertToDefinitions(mappedTypeDef *, classDef *, FILE *); +static void generateEncodedClass(sipSpec *, classDef *, int, FILE *); +static int generateArgParser(sipSpec *, signatureDef *, classDef *, ctorDef *, + overDef *, int, FILE *); +static void generateTry(throwArgs *, FILE *); +static void generateCatch(throwArgs *ta, signatureDef *sd, FILE *fp); +static void generateThrowSpecifier(throwArgs *, FILE *); +static void generateSlot(sipSpec *pt, classDef *cd, enumDef *ed, memberDef *md, + FILE *fp); +static void generateCastZero(argDef *ad, FILE *fp); +static void generateCallDefaultCtor(ctorDef *ct, FILE *fp); +static int countVirtuals(classDef *); +static int skipOverload(overDef *, memberDef *, classDef *, classDef *, int); +static int compareMethTab(const void *, const void *); +static int compareEnumMembers(const void *, const void *); +static char *getSubFormatChar(char, argDef *); +static char *createIfaceFileName(char *, ifaceFileDef *, char *); +static FILE *createCompilationUnit(sipSpec *pt, char *fname, + char *description); +static FILE *createFile(sipSpec *, char *, char *); +static void closeFile(FILE *); +static void prScopedName(FILE *fp, scopedNameDef *snd, char *sep); +static void prTypeName(FILE *, argDef *, int); +static void prScopedClassName(FILE *, classDef *, char *); +static int isZeroArgSlot(memberDef *md); +static int isMultiArgSlot(memberDef *md); +static int isIntArgSlot(memberDef *md); +static int isInplaceNumberSlot(memberDef *md); +static int isInplaceSequenceSlot(memberDef *md); +static int needErrorFlag(codeBlock *cb); +static int needNewInstance(argDef *ad); +static int needDealloc(classDef *cd); +static char getBuildResultFormat(argDef *ad); +static const char *getParseResultFormat(argDef *ad, int isres, int xfervh); +static void generateParseResultExtraArgs(argDef *ad, int isres, FILE *fp); +static char *makePartName(char *codeDir, char *mname, int part, + char *srcSuffix); +static void normaliseArgs(signatureDef *); +static void restoreArgs(signatureDef *); +static const char *slotName(slotType st); +static void ints_intro(classDef *cd, FILE *fp); +static const char *argName(const char *name, codeBlock *cb); +static int usedInCode(codeBlock *code, const char *str); +static void generateDefaultValue(argDef *ad, int argnr, FILE *fp); +static void generateClassFromVoid(classDef *cd, const char *cname, + const char *vname, FILE *fp); +static void generateMappedTypeFromVoid(mappedTypeDef *mtd, const char *cname, + const char *vname, FILE *fp); +static int generateSubClassConvertors(sipSpec *pt, FILE *fp); +static void generateRegisterMetaType(classDef *cd, FILE *fp); + + +/* + * Generate the code from a specification. + */ +void generateCode(sipSpec *pt, char *codeDir, char *buildfile, char *docFile, + char *srcSuffix, int except, int trace, int releaseGIL, int parts, + stringList *xsl) +{ + exceptions = except; + tracing = trace; + release_gil = releaseGIL; + generating_c = pt->genc; + + if (srcSuffix == NULL) + srcSuffix = (generating_c ? ".c" : ".cpp"); + + /* Generate the documentation. */ + if (docFile != NULL) + generateDocumentation(pt,docFile); + + /* Generate the code. */ + if (codeDir != NULL) + { + generateCpp(pt,codeDir,srcSuffix,&parts); + generateInternalAPIHeader(pt,codeDir,xsl); + } + + /* Generate the build file. */ + if (buildfile != NULL) + generateBuildFile(pt,buildfile,srcSuffix,parts); +} + + +/* + * Generate the documentation. + */ +static void generateDocumentation(sipSpec *pt, char *docFile) +{ + FILE *fp; + codeBlock *cb; + + fp = createFile(pt, docFile, NULL); + + for (cb = pt->docs; cb != NULL; cb = cb->next) + fputs(cb->frag, fp); + + closeFile(fp); +} + + +/* + * Generate the build file. + */ +static void generateBuildFile(sipSpec *pt, char *buildFile, char *srcSuffix, + int parts) +{ + char *mname = pt->module->name; + ifaceFileDef *iff; + FILE *fp; + + fp = createFile(pt, buildFile, NULL); + + prcode(fp, "target = %s\nsources = ", mname); + + if (parts) + { + int p; + + for (p = 0; p < parts; ++p) + { + if (p > 0) + prcode(fp, " "); + + prcode(fp, "sip%spart%d%s", mname, p, srcSuffix); + } + } + else + { + prcode(fp, "sip%scmodule%s", mname, srcSuffix); + + for (iff = pt->ifacefiles; iff != NULL; iff = iff->next) + { + if (iff->module != pt->module) + continue; + + if (iff->type == exception_iface) + continue; + + prcode(fp, " sip%s%F%s", mname, iff->fqcname, srcSuffix); + } + } + + prcode(fp, "\nheaders = sipAPI%s.h", mname); + + for (iff = pt->ifacefiles; iff != NULL; iff = iff->next) + { + char *imname; + + imname = (iff->module == pt->module ? mname : iff->module->name); + + prcode(fp, " sip%s%F.h", imname, iff->fqcname); + } + + prcode(fp, "\n"); + + closeFile(fp); +} + + +/* + * Generate an expression in C++. + */ +void generateExpression(valueDef *vd, FILE *fp) +{ + while (vd != NULL) + { + if (vd->vunop != '\0') + prcode(fp,"%c",vd->vunop); + + switch (vd->vtype) + { + case qchar_value: + prcode(fp,"'%c'",vd->u.vqchar); + break; + + case string_value: + prcode(fp,"\"%s\"",vd->u.vstr); + break; + + case numeric_value: + prcode(fp,"%l",vd->u.vnum); + break; + + case real_value: + prcode(fp,"%g",vd->u.vreal); + break; + + case scoped_value: + if (prcode_xml) + prScopedName(fp, vd->u.vscp, "."); + else + prcode(fp, "%S", vd->u.vscp); + + break; + + case fcall_value: + generateSimpleFunctionCall(vd->u.fcd,fp); + break; + } + + if (vd->vbinop != '\0') + prcode(fp," %c ",vd->vbinop); + + vd = vd->next; + } +} + + +/* + * Generate the C++ internal module API header file. + */ +static void generateInternalAPIHeader(sipSpec *pt,char *codeDir,stringList *xsl) +{ + char *hfile, *mname = pt->module->name; + int noIntro; + FILE *fp; + nameDef *nd; + moduleDef *mod; + moduleListDef *mld; + + hfile = concat(codeDir,"/sipAPI",mname,".h",NULL); + fp = createFile(pt,hfile,"Internal module API header file."); + + /* Include files. */ + + prcode(fp, +"\n" +"#ifndef _%sAPI_H\n" +"#define _%sAPI_H\n" +"\n" +"\n" +"#include <sip.h>\n" + ,mname + ,mname); + + if (optRegisterTypes(pt)) + prcode(fp, +"\n" +"#include <QMetaType>\n" + ); + + /* Define the enabled features. */ + noIntro = TRUE; + + for (mod = pt->modules; mod != NULL; mod = mod->next) + { + qualDef *qd; + + for (qd = mod->qualifiers; qd != NULL; qd = qd->next) + if (qd->qtype == feature_qualifier && !excludedFeature(xsl,qd)) + { + if (noIntro) + { + prcode(fp, +"\n" +"\n" +"/* These are the features that are enabled. */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +"#define SIP_FEATURE_%s\n" + ,qd->name); + } + } + + generateCppCodeBlock(pt->exphdrcode,fp); + generateCppCodeBlock(pt->hdrcode,fp); + + /* Shortcuts that hide the messy detail of the APIs. */ + noIntro = TRUE; + + for (nd = pt->namecache; nd != NULL; nd = nd->next) + { + if (!isClassName(nd)) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +"\n" +"/*\n" +" * Convenient names to refer to the names of classes defined in this module.\n" +" * These are part of the public API.\n" +" */\n" +"\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +"#define sipName_%s %N\n" + ,nd->text,nd); + } + + prcode(fp, +"\n" +"\n" +"/* Convenient names to call the SIP API. */\n" +"#define sipConvertFromSliceObject(o,len,start,stop,step,slen) PySlice_GetIndicesEx((PySliceObject *)(o),(len),(start),(stop),(step),(slen))\n" +"#define sipIsSubClassInstance(o,wt) PyObject_TypeCheck((o),(PyTypeObject *)(wt))\n" +"\n" +"#define sipMapStringToClass sipAPI_%s->api_map_string_to_class\n" +"#define sipMapIntToClass sipAPI_%s->api_map_int_to_class\n" +"#define sipMalloc sipAPI_%s->api_malloc\n" +"#define sipFree sipAPI_%s->api_free\n" +"#define sipBuildResult sipAPI_%s->api_build_result\n" +"#define sipCallMethod sipAPI_%s->api_call_method\n" +"#define sipParseResult sipAPI_%s->api_parse_result\n" +"#define sipParseArgs sipAPI_%s->api_parse_args\n" +"#define sipParsePair sipAPI_%s->api_parse_pair\n" +"#define sipCommonCtor sipAPI_%s->api_common_ctor\n" +"#define sipCommonDtor sipAPI_%s->api_common_dtor\n" +"#define sipConvertFromSequenceIndex sipAPI_%s->api_convert_from_sequence_index\n" +"#define sipConvertFromVoidPtr sipAPI_%s->api_convert_from_void_ptr\n" +"#define sipConvertToCpp sipAPI_%s->api_convert_to_cpp\n" +"#define sipConvertToVoidPtr sipAPI_%s->api_convert_to_void_ptr\n" +"#define sipNoFunction sipAPI_%s->api_no_function\n" +"#define sipNoMethod sipAPI_%s->api_no_method\n" +"#define sipAbstractMethod sipAPI_%s->api_abstract_method\n" +"#define sipBadClass sipAPI_%s->api_bad_class\n" +"#define sipBadSetType sipAPI_%s->api_bad_set_type\n" +"#define sipBadCatcherResult sipAPI_%s->api_bad_catcher_result\n" +"#define sipBadOperatorArg sipAPI_%s->api_bad_operator_arg\n" +"#define sipTrace sipAPI_%s->api_trace\n" +"#define sipTransfer sipAPI_%s->api_transfer\n" +"#define sipTransferBack sipAPI_%s->api_transfer_back\n" +"#define sipTransferTo sipAPI_%s->api_transfer_to\n" +"#define sipWrapper_Check sipAPI_%s->api_wrapper_check\n" +"#define sipGetWrapper sipAPI_%s->api_get_wrapper\n" +"#define sipGetCppPtr sipAPI_%s->api_get_cpp_ptr\n" +"#define sipGetComplexCppPtr sipAPI_%s->api_get_complex_cpp_ptr\n" +"#define sipIsPyMethod sipAPI_%s->api_is_py_method\n" +"#define sipCallHook sipAPI_%s->api_call_hook\n" +"#define sipStartThread sipAPI_%s->api_start_thread\n" +"#define sipEndThread sipAPI_%s->api_end_thread\n" +"#define sipEmitSignal sipAPI_%s->api_emit_signal\n" +"#define sipConnectRx sipAPI_%s->api_connect_rx\n" +"#define sipDisconnectRx sipAPI_%s->api_disconnect_rx\n" +"#define sipGetSender sipAPI_%s->api_get_sender\n" +"#define sipRaiseUnknownException sipAPI_%s->api_raise_unknown_exception\n" +"#define sipRaiseClassException sipAPI_%s->api_raise_class_exception\n" +"#define sipRaiseSubClassException sipAPI_%s->api_raise_sub_class_exception\n" +"#define sipBadLengthForSlice sipAPI_%s->api_bad_length_for_slice\n" +"#define sipClassName sipAPI_%s->api_class_name\n" +"#define sipAddClassInstance sipAPI_%s->api_add_class_instance\n" +"#define sipAddMappedTypeInstance sipAPI_%s->api_add_mapped_type_instance\n" +"#define sipAddEnumInstance sipAPI_%s->api_add_enum_instance\n" +"#define sipConvertFromNamedEnum sipAPI_%s->api_convert_from_named_enum\n" +"#define sipGetAddress sipAPI_%s->api_get_address\n" +"#define sipFreeConnection sipAPI_%s->api_free_connection\n" +"#define sipEmitToSlot sipAPI_%s->api_emit_to_slot\n" +"#define sipSameConnection sipAPI_%s->api_same_connection\n" +"#define sipPySlotExtend sipAPI_%s->api_pyslot_extend\n" +"#define sipConvertRx sipAPI_%s->api_convert_rx\n" +"#define sipAddDelayedDtor sipAPI_%s->api_add_delayed_dtor\n" +"#define sipCanConvertToInstance sipAPI_%s->api_can_convert_to_instance\n" +"#define sipCanConvertToMappedType sipAPI_%s->api_can_convert_to_mapped_type\n" +"#define sipConvertToInstance sipAPI_%s->api_convert_to_instance\n" +"#define sipConvertToMappedType sipAPI_%s->api_convert_to_mapped_type\n" +"#define sipForceConvertToInstance sipAPI_%s->api_force_convert_to_instance\n" +"#define sipForceConvertToMappedType sipAPI_%s->api_force_convert_to_mapped_type\n" +"#define sipReleaseInstance sipAPI_%s->api_release_instance\n" +"#define sipReleaseMappedType sipAPI_%s->api_release_mapped_type\n" +"#define sipConvertFromInstance sipAPI_%s->api_convert_from_instance\n" +"#define sipConvertFromNewInstance sipAPI_%s->api_convert_from_new_instance\n" +"#define sipConvertFromMappedType sipAPI_%s->api_convert_from_mapped_type\n" +"#define sipGetState sipAPI_%s->api_get_state\n" +"#define sipFindMappedType sipAPI_%s->api_find_mapped_type\n" +"#define sipLong_AsUnsignedLong sipAPI_%s->api_long_as_unsigned_long\n" +"#define sipExportSymbol sipAPI_%s->api_export_symbol\n" +"#define sipImportSymbol sipAPI_%s->api_import_symbol\n" +"#define sipRegisterIntTypes sipAPI_%s->api_register_int_types\n" +"#define sipParseSignature sipAPI_%s->api_parse_signature\n" +"#define sipFindClass sipAPI_%s->api_find_class\n" +"#define sipFindNamedEnum sipAPI_%s->api_find_named_enum\n" +"#define sipString_AsChar sipAPI_%s->api_string_as_char\n" +"#define sipUnicode_AsWChar sipAPI_%s->api_unicode_as_wchar\n" +"#define sipUnicode_AsWString sipAPI_%s->api_unicode_as_wstring\n" + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname + ,mname); + + /* The name strings. */ + noIntro = TRUE; + + for (nd = pt->namecache; nd != NULL; nd = nd->next) + { + if (!isUsedName(nd)) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +"\n" +"/* The strings used by this module. */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +"extern char %N[];\n" + ,nd); + } + + /* The unscoped enum macros. */ + generateEnumMacros(pt, NULL, fp); + + prcode(fp, +"\n" +"\n" +"/* The SIP API, this module's API and the APIs of any imported modules. */\n" +"extern const sipAPIDef *sipAPI_%s;\n" +"extern sipExportedModuleDef sipModuleAPI_%s;\n" + ,mname + ,mname,mname); + + for (mld = pt->module->allimports; mld != NULL; mld = mld->next) + prcode(fp, +"extern const sipExportedModuleDef *sipModuleAPI_%s_%s;\n" + ,mname,mld->module->name); + + if (optQ_OBJECT4(pt)) + prcode(fp, +"\n" +"typedef const QMetaObject *(*sip_qt_metaobject_func)(sipWrapper *,sipWrapperType *,const QMetaObject *);\n" +"extern sip_qt_metaobject_func sip_%s_qt_metaobject;\n" +"\n" +"typedef int (*sip_qt_metacall_func)(sipWrapper *,sipWrapperType *,QMetaObject::Call,int,void **);\n" +"extern sip_qt_metacall_func sip_%s_qt_metacall;\n" + , mname + , mname); + + /* + * Note that we don't forward declare the virtual handlers. This is + * because we would need to #include everything needed for their + * argument types. + */ + + prcode(fp, +"\n" +"#endif\n" + ); + + closeFile(fp); + free(hfile); +} + + +/* + * Return the filename of a source code part on the heap. + */ +static char *makePartName(char *codeDir,char *mname,int part,char *srcSuffix) +{ + char buf[20]; + + sprintf(buf,"part%d",part); + + return concat(codeDir,"/sip",mname,buf,srcSuffix,NULL); +} + + +/* + * Generate the C/C++ code. + */ +static void generateCpp(sipSpec *pt, char *codeDir, char *srcSuffix, int *parts) +{ + char *cppfile, *mname = pt->module->name; + int noIntro, nrSccs = 0, files_in_part, max_per_part, this_part; + int is_inst_class, is_inst_voidp, is_inst_char, is_inst_string; + int is_inst_int, is_inst_long, is_inst_ulong, is_inst_longlong; + int is_inst_ulonglong, is_inst_double, is_inst_enum, nr_enummembers; + int hasexternal = FALSE, slot_extenders = FALSE, ctor_extenders = FALSE; + FILE *fp; + moduleListDef *mld; + classDef *cd; + memberDef *md; + ifaceFileDef *iff; + virtHandlerDef *vhd; + nameDef *nd; + exceptionDef *xd; + + /* Calculate the number of files in each part. */ + if (*parts) + { + int nr_files = 1; + + for (iff = pt->ifacefiles; iff != NULL; iff = iff->next) + if (iff->module == pt->module) + ++nr_files; + + max_per_part = (nr_files + *parts - 1) / *parts; + files_in_part = 1; + this_part = 0; + + cppfile = makePartName(codeDir,mname,0,srcSuffix); + } + else + cppfile = concat(codeDir,"/sip",mname,"cmodule",srcSuffix,NULL); + + fp = createCompilationUnit(pt, cppfile, "Module code."); + + prcode(fp, +"\n" +"#include \"sipAPI%s.h\"\n" +"\n" + ,mname); + + for (iff = pt->ifacefiles; iff != NULL; iff = iff->next) + if (iff->module == pt->module && iff->type != exception_iface) + prcode(fp, +"#include \"sip%s%F.h\"\n" + ,iff->module->name,iff->fqcname); + + generateUsedIncludes(pt->used, FALSE, fp); + + /* + * If there should be a Qt support API then generate stubs values for the + * optional parts. These should be undefined in %ModuleCode if a C++ + * implementation is provided. + */ + if (pt->qobjclass >= 0) + prcode(fp, +"\n" +"#define sipQtIsQtSignal 0\n" +"#define sipQtCreateUniversalSignalShortcut 0\n" +"#define sipQtCreateUniversalSignal 0\n" +"#define sipQtFindUniversalSignalShortcut 0\n" +"#define sipQtFindUniversalSignal 0\n" +"#define sipQtEmitSignalShortcut 0\n" +"#define sipQtEmitSignal 0\n" + ); + + /* Define the names. */ + noIntro = TRUE; + + for (nd = pt->namecache; nd != NULL; nd = nd->next) + { + if (!isUsedName(nd)) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +"\n" +"/* Define the strings used by this module. */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +"char %N[] = \"%s\";\n" + ,nd,nd->text); + } + + /* Generate the C++ code blocks. */ + generateCppCodeBlock(pt->cppcode,fp); + + /* Generate any virtual handler declarations. */ + for (vhd = pt->module->virthandlers; vhd != NULL; vhd = vhd->next) + if (!isDuplicateVH(vhd)) + generateVirtualHandler(pt,vhd,fp); + + /* Generate the global functions. */ + for (md = pt->othfuncs; md != NULL; md = md->next) + { + if (md->module != pt->module) + continue; + + if (md->slot == no_slot) + generateOrdinaryFunction(pt,NULL,md,fp); + else + { + overDef *od; + + /* + * Make sure that there is still an overload and we + * haven't moved them all to classes. + */ + for (od = pt->overs; od != NULL; od = od->next) + if (od->common == md) + { + generateSlot(pt, NULL, NULL, md, fp); + slot_extenders = TRUE; + break; + } + } + } + + /* Generate any class specific ctor or slot extenders. */ + for (cd = pt->proxies; cd != NULL; cd = cd->next) + { + if (cd->ctors != NULL) + { + generateTypeInit(pt, cd, fp); + ctor_extenders = TRUE; + } + + for (md = cd->members; md != NULL; md = md->next) + { + generateSlot(pt, cd, NULL, md, fp); + slot_extenders = TRUE; + } + } + + /* Generate any ctor extender table. */ + if (ctor_extenders) + { + prcode(fp, +"\n" +"static sipInitExtenderDef initExtenders[] = {\n" + ); + + for (cd = pt->proxies; cd != NULL; cd = cd->next) + if (cd->ctors != NULL) + { + prcode(fp, +" {init_%C, ", classFQCName(cd)); + + generateEncodedClass(pt, cd, 0, fp); + + prcode(fp, ", NULL},\n" + ); + } + + prcode(fp, +" {NULL, {0, 0, 0}, NULL}\n" +"};\n" + ); + } + + /* Generate any slot extender table. */ + if (slot_extenders) + { + prcode(fp, +"\n" +"static sipPySlotExtenderDef slotExtenders[] = {\n" + ); + + for (md = pt->othfuncs; md != NULL; md = md->next) + { + overDef *od; + + if (md->module != pt->module) + continue; + + if (md->slot == no_slot) + continue; + + for (od = pt->overs; od != NULL; od = od->next) + if (od->common == md) + { + prcode(fp, +" {(void *)slot_%s, %s, {0, 0, 0}},\n" + , md->pyname->text, slotName(md->slot)); + + break; + } + } + + for (cd = pt->proxies; cd != NULL; cd = cd->next) + for (md = cd->members; md != NULL; md = md->next) + { + prcode(fp, +" {(void *)slot_%C_%s, %s, ", classFQCName(cd), md->pyname->text, slotName(md->slot)); + + generateEncodedClass(pt, cd, 0, fp); + + prcode(fp, "},\n" + ); + } + + prcode(fp, +" {NULL, (sipPySlotType)0, {0, 0, 0}}\n" +"};\n" + ); + } + + /* Generate the global access functions. */ + generateAccessFunctions(pt,NULL,fp); + + /* Generate the module data structures. */ + if (pt->module->nrclasses > 0) + { + nrSccs = generateSubClassConvertors(pt, fp); + + prcode(fp, +"\n" +"\n" +"/*\n" +" * This defines each class in this module. The values are replaced by the\n" +" * proper Python types during the export process.\n" +" */\n" +"static sipWrapperType *typesTable[] = {\n" + ); + + generateClassTableEntries(pt, &pt->module->root, fp); + + prcode(fp, +"};\n" + ); + + /* Generate the external classes table if needed. */ + for (cd = pt->classes; cd != NULL; cd = cd->next) + { + if (!isExternal(cd)) + continue; + + if (cd->iff->module != pt->module) + continue; + + if (!hasexternal) + { + prcode(fp, +"\n" +"\n" +"/* This defines each external type declared in this module, */\n" +"static sipExternalTypeDef externalTypesTable[] = {\n" + ); + + hasexternal = TRUE; + } + + prcode(fp, +" {%d, \"", cd->classnr); + prScopedName(fp, classFQCName(cd), "."); + prcode(fp,"\"},\n" + ); + } + + if (hasexternal) + prcode(fp, +" {-1, NULL}\n" +"};\n" + ); + } + + if (pt->module->nrmappedtypes > 0) + { + mappedTypeDef *mtd; + argDef type; + + type.argflags = 0; + type.name = NULL; + type.nrderefs = 0; + type.defval = NULL; + + prcode(fp, +"\n" +"\n" +"/* This defines each mapped type in this module. */\n" +"static sipMappedType *mappedTypesTable[] = {\n" + ); + + for (mtd = pt->mappedtypes; mtd != NULL; mtd = mtd->next) + { + if (mtd->iff->module != pt->module) + continue; + + type.atype = mapped_type; + type.u.mtd = mtd; + + prcode(fp, +" &sipMappedTypeDef_%T,\n" + ,&type); + } + + prcode(fp, +" 0\n" +"};\n" + ); + } + + if (pt->module->nrenums > 0) + { + enumDef *ed; + + prcode(fp, +"\n" +"\n" +"/* Define each named enum in this module. */\n" + ); + + /* Generate any slot tables. */ + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + memberDef *slot; + + if (ed->module != pt->module || ed->fqcname == NULL) + continue; + + if (ed->slots == NULL) + continue; + + for (slot = ed->slots; slot != NULL; slot = slot->next) + generateSlot(pt, NULL, ed, slot, fp); + + prcode(fp, +"\n" +"static sipPySlotDef slots_%C[] = {\n" + , ed->fqcname); + + for (slot = ed->slots; slot != NULL; slot = slot->next) + { + const char *stype; + + if ((stype = slotName(slot->slot)) != NULL) + prcode(fp, +" {(void *)slot_%C_%s, %s},\n" + , ed->fqcname, slot->pyname->text, stype); + } + + prcode(fp, +" {0, (sipPySlotType)0}\n" +"};\n" +"\n" + ); + } + + prcode(fp, +"static sipEnumDef enumTypesTable[] = {\n" + ); + + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + char *emname; + + if (ed->module != pt->module || ed->fqcname == NULL) + continue; + + if (ed->ecd == NULL) + emname = mname; + else if (ed->ecd->real == NULL) + emname = ed->module->name; + else + emname = ed->ecd->real->iff->module->name; + + prcode(fp, +" {\"%s.%P\", ", emname, ed->ecd, ed->pyname->text); + + if (isRenamedEnum(ed) || (ed->ecd != NULL && isRenamedClass(ed->ecd))) + prcode(fp, "\"%S\", ", ed->fqcname); + else + prcode(fp, "NULL, "); + + if (ed->ecd == NULL) + prcode(fp, "-1"); + else + prcode(fp, "%d", ed->ecd->classnr); + + if (ed->slots != NULL) + prcode(fp, ", slots_%C", ed->fqcname); + else + prcode(fp, ", NULL"); + + prcode(fp, "},\n" + ); + } + + prcode(fp, +"};\n" + ); + + nr_enummembers = generateEnumMemberTable(pt,NULL,fp); + } + else + nr_enummembers = 0; + + if (pt->module->nrtypedefs > 0) + { + typedefDef *td; + + prcode(fp, +"\n" +"\n" +"/*\n" +" * These define each typedef in this module. They are only needed in case\n" +" * they are used as arguments to Qt signals.\n" +" */\n" +"\n" +"static sipTypedefDef typedefsTable[] = {\n" + ); + + for (td = pt->typedefs; td != NULL; td = td->next) + { + char *tdmname, *sat; + scopedNameDef *fqname; + + if (td->module != pt->module) + continue; + + fqname = NULL; + tdmname = NULL; + sat = "unknown"; + + switch (td->type.atype) + { + case string_type: + sat = (td->type.nrderefs == 0 ? "char" : "string"); + break; + + case sstring_type: + sat = (td->type.nrderefs == 0 ? "schar" : "sstring"); + break; + + case ustring_type: + sat = (td->type.nrderefs == 0 ? "uchar" : "ustring"); + break; + + case wstring_type: + sat = (td->type.nrderefs == 0 ? "wchar" : "wstring"); + break; + + case short_type: + sat = "short"; + break; + + case ushort_type: + sat = "ushort"; + break; + + case cint_type: + case int_type: + sat = "int"; + break; + + case uint_type: + sat = "uint"; + break; + + case long_type: + sat = "long"; + break; + + case ulong_type: + sat = "ulong"; + break; + + case longlong_type: + sat = "longlong"; + break; + + case ulonglong_type: + sat = "ulonglong"; + break; + + case cfloat_type: + case float_type: + sat = "float"; + break; + + case cdouble_type: + case double_type: + sat = "double"; + break; + + case bool_type: + case cbool_type: + sat = "bool"; + break; + + case void_type: + if (td->type.nrderefs != 0) + sat = "void"; + break; + + case enum_type: + if ((fqname = td->type.u.ed->fqcname) != NULL) + { + sat = "enum"; + + if (td->type.u.ed->module != pt->module) + tdmname = td->type.u.ed->module->fullname; + } + break; + + case class_type: + sat = "class"; + fqname = classFQCName(td->type.u.cd); + + if (td->type.u.cd->iff->module != pt->module) + tdmname = td->type.u.cd->iff->module->fullname; + break; + + case mapped_type: + sat = "mtype"; + fqname = td->type.u.mtd->iff->fqcname; + + if (td->type.u.mtd->iff->module != pt->module) + tdmname = td->type.u.mtd->iff->module->fullname; + break; + } + + prcode(fp, +" {\"%S\", %s_sat", td->fqname, sat); + + if (fqname != NULL) + prcode(fp, ", \"%S\"", fqname); + else + prcode(fp, ", NULL"); + + if (tdmname != NULL) + prcode(fp, ", \"%s\"", tdmname); + else + prcode(fp, ", NULL"); + + prcode(fp, "},\n" + ); + } + + prcode(fp, +" {NULL, unknown_sat, NULL, NULL}\n" +"};\n" + ); + } + + if (pt->module->nrvirthandlers > 0) + { + prcode(fp, +"\n" +"\n" +"/*\n" +" * This defines the virtual handlers that this module implements and can be\n" +" * used by other modules.\n" +" */\n" +"static sipVirtHandlerFunc virtHandlersTable[] = {\n" + ); + + for (vhd = pt->module->virthandlers; vhd != NULL; vhd = vhd->next) + if (!isDuplicateVH(vhd)) + prcode(fp, +" (sipVirtHandlerFunc)sipVH_%s_%d,\n" + ,mname,vhd->virthandlernr); + + prcode(fp, +"};\n" + ); + } + + if (pt->module->allimports != NULL) + { + prcode(fp, +"\n" +"\n" +"/* This defines the modules that this module needs to import. */\n" +"static sipImportedModuleDef importsTable[] = {\n" + ); + + for (mld = pt->module->allimports; mld != NULL; mld = mld->next) + prcode(fp, +" {\"%s\", %d, NULL},\n" + , mld->module->fullname, mld->module->version); + + prcode(fp, +" {NULL, -1, NULL}\n" +"};\n" + ); + } + + if (nrSccs > 0) + { + prcode(fp, +"\n" +"\n" +"/* This defines the class sub-convertors that this module defines. */\n" +"static sipSubClassConvertorDef convertorsTable[] = {\n" + ); + + for (cd = pt->classes; cd != NULL; cd = cd->next) + { + if (cd->iff->module != pt->module) + continue; + + if (cd->convtosubcode == NULL) + continue; + + prcode(fp, +" {sipSubClass_%C, ",classFQCName(cd)); + + generateEncodedClass(pt,cd->subbase,0,fp); + + prcode(fp,", NULL},\n"); + } + + prcode(fp, +" {NULL, {0, 0, 0}, NULL}\n" +"};\n" + ); + } + + /* Generate any license information. */ + if (pt->module->license != NULL) + { + licenseDef *ld = pt->module->license; + + prcode(fp, +"\n" +"\n" +"/* Define the module's license. */\n" +"static sipLicenseDef module_license = {\n" + ); + + prcode(fp, +" \"%s\",\n" + ,ld->type); + + if (ld->licensee != NULL) + prcode(fp, +" \"%s\",\n" + ,ld->licensee); + else + prcode(fp, +" NULL,\n" + ); + + if (ld->timestamp != NULL) + prcode(fp, +" \"%s\",\n" + ,ld->timestamp); + else + prcode(fp, +" NULL,\n" + ); + + if (ld->sig != NULL) + prcode(fp, +" \"%s\"\n" + ,ld->sig); + else + prcode(fp, +" NULL\n" + ); + + prcode(fp, +"};\n" + ); + } + + /* Generate each instance table. */ + is_inst_class = generateClasses(pt,NULL,fp); + is_inst_voidp = generateVoidPointers(pt,NULL,fp); + is_inst_char = generateChars(pt,NULL,fp); + is_inst_string = generateStrings(pt,NULL,fp); + is_inst_int = generateInts(pt,NULL,fp); + is_inst_long = generateLongs(pt,NULL,fp); + is_inst_ulong = generateUnsignedLongs(pt,NULL,fp); + is_inst_longlong = generateLongLongs(pt,NULL,fp); + is_inst_ulonglong = generateUnsignedLongLongs(pt,NULL,fp); + is_inst_double = generateDoubles(pt,NULL,fp); + is_inst_enum = generateEnums(pt,NULL,fp); + + /* Generate any exceptions table. */ + if (pt->module->nrexceptions > 0) + prcode(fp, +"\n" +"\n" +"static PyObject *exceptionsTable[%d];\n" + ,pt->module->nrexceptions); + + /* Generate any Qt support API. */ + if (pt->qobjclass >= 0) + prcode(fp, +"\n" +"\n" +"/* This defines the Qt support API. */\n" +"\n" +"static sipQtAPI qtAPI = {\n" +" &typesTable[%d],\n" +" sipQtIsQtSignal,\n" +" sipQtCreateUniversalSignalShortcut,\n" +" sipQtCreateUniversalSignal,\n" +" sipQtFindUniversalSignalShortcut,\n" +" sipQtFindUniversalSignal,\n" +" sipQtEmitSignalShortcut,\n" +" sipQtEmitSignal,\n" +" sipQtCreateUniversalSlot,\n" +" sipQtDestroyUniversalSlot,\n" +" sipQtFindSlot,\n" +" sipQtConnect,\n" +" sipQtDisconnect,\n" +" sipQtSignalsBlocked,\n" +" sipQtGetSender,\n" +" sipQtForgetSender,\n" +" sipQtSameSignalSlotName,\n" +" sipQtFindConnection\n" +"};\n" + ,pt->qobjclass); + + prcode(fp, +"\n" +"\n" +"/* This defines this module. */\n" +"sipExportedModuleDef sipModuleAPI_%s = {\n" +" NULL,\n" +" SIP_API_MINOR_NR,\n" +" \"%s\",\n" +" NULL,\n" +" %d,\n" +" %s,\n" +" %s,\n" +" %d,\n" +" %s,\n" +" %s,\n" +" %s,\n" +" %d,\n" +" NULL,\n" +" %s,\n" +" %d,\n" +" %s,\n" +" %s,\n" +" %s,\n" +" %s,\n" +" {%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s},\n" +" %s,\n" +" %s,\n" +" %s,\n" +" %s,\n" +" %s,\n" +" NULL\n" +"};\n" + , mname + , pt->module->fullname + , pt->module->version + , pt->module->allimports != NULL ? "importsTable" : "NULL" + , pt->qobjclass >= 0 ? "&qtAPI" : "NULL" + , pt->module->nrclasses + , pt->module->nrclasses > 0 ? "typesTable" : "NULL" + , hasexternal ? "externalTypesTable" : "NULL" + , pt->module->nrmappedtypes > 0 ? "mappedTypesTable" : "NULL" + , pt->module->nrenums + , pt->module->nrenums > 0 ? "enumTypesTable" : "NULL" + , nr_enummembers + , nr_enummembers > 0 ? "enummembers" : "NULL" + , pt->module->nrtypedefs > 0 ? "typedefsTable" : "NULL" + , pt->module->nrvirthandlers > 0 ? "virtHandlersTable" : "NULL" + , nrSccs > 0 ? "convertorsTable" : "NULL" + , is_inst_class ? "classInstances" : "NULL" + , is_inst_voidp ? "voidPtrInstances" : "NULL" + , is_inst_char ? "charInstances" : "NULL" + , is_inst_string ? "stringInstances" : "NULL" + , is_inst_int ? "intInstances" : "NULL" + , is_inst_long ? "longInstances" : "NULL" + , is_inst_ulong ? "unsignedLongInstances" : "NULL" + , is_inst_longlong ? "longLongInstances" : "NULL" + , is_inst_ulonglong ? "unsignedLongLongInstances" : "NULL" + , is_inst_double ? "doubleInstances" : "NULL" + , is_inst_enum ? "enumInstances" : "NULL" + , pt->module->license != NULL ? "&module_license" : "NULL" + , pt->module->nrexceptions > 0 ? "exceptionsTable" : "NULL" + , slot_extenders ? "slotExtenders" : "NULL" + , ctor_extenders ? "initExtenders" : "NULL" + , hasDelayedDtors(pt->module) ? "sipDelayedDtors" : "NULL"); + + /* Generate the storage for the external API pointers. */ + prcode(fp, +"\n" +"\n" +"/* The SIP API and the APIs of any imported modules. */\n" +"const sipAPIDef *sipAPI_%s;\n" + ,mname); + + for (mld = pt->module->allimports; mld != NULL; mld = mld->next) + prcode(fp, +"const sipExportedModuleDef *sipModuleAPI_%s_%s;\n" + ,mname,mld->module->name); + + if (optQ_OBJECT4(pt)) + prcode(fp, +"\n" +"sip_qt_metaobject_func sip_%s_qt_metaobject;\n" +"sip_qt_metacall_func sip_%s_qt_metacall;\n" + , mname + , mname); + + /* Generate the Python module initialisation function. */ + prcode(fp, +"\n" +"\n" +"/* The Python module initialisation function. */\n" +"#if defined(SIP_STATIC_MODULE)\n" +"%svoid init%s()\n" +"#else\n" +"PyMODINIT_FUNC init%s()\n" +"#endif\n" +"{\n" +" static PyMethodDef sip_methods[] = {\n" + ,(generating_c ? "" : "extern \"C\" "), mname + ,mname); + + /* Generate the global functions. */ + + for (md = pt->othfuncs; md != NULL; md = md->next) + if (md->module == pt->module && md->slot == no_slot) + prcode(fp, +" {%N, func_%s, METH_VARARGS, NULL},\n" + ,md->pyname,md->pyname->text); + + prcode(fp, +" {0, 0, 0, 0}\n" +" };\n" +"\n" +" PyObject *sipModule, *sipModuleDict, *sip_sipmod, *sip_capiobj;\n" +"\n" + ); + + /* Generate the pre-initialisation code. */ + generateCppCodeBlock(pt->preinitcode,fp); + + prcode(fp, +" /* Initialise the module and get it's dictionary. */\n" +" sipModule = Py_InitModule((char *)sipModuleAPI_%s.em_name,sip_methods);\n" +" sipModuleDict = PyModule_GetDict(sipModule);\n" +"\n" +" /* Import the SIP module and get it's API. */\n" +" sip_sipmod = PyImport_ImportModule((char *)\"sip\");\n" +"\n" +" if (sip_sipmod == NULL)\n" +" return;\n" +"\n" +" sip_capiobj = PyDict_GetItemString(PyModule_GetDict(sip_sipmod),\"_C_API\");\n" +"\n" +" if (sip_capiobj == NULL || !PyCObject_Check(sip_capiobj))\n" +" return;\n" +"\n" + ,mname); + + if (generating_c) + prcode(fp, +" sipAPI_%s = (const sipAPIDef *)PyCObject_AsVoidPtr(sip_capiobj);\n" + ,mname); + else + prcode(fp, +" sipAPI_%s = reinterpret_cast<const sipAPIDef *>(PyCObject_AsVoidPtr(sip_capiobj));\n" + ,mname); + + prcode(fp, +"\n" +" /* Export the module and publish it's API. */\n" +" if (sipAPI_%s->api_export_module(&sipModuleAPI_%s,SIP_API_MAJOR_NR,SIP_API_MINOR_NR,sipModuleDict) < 0)\n" +" return;\n" + ,mname + ,mname); + + noIntro = TRUE; + + for (mld = pt->module->allimports; mld != NULL; mld = mld->next) + { + if (noIntro) + { + prcode(fp, +"\n" +" /* Get the APIs of the modules that this one is dependent on. */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" sipModuleAPI_%s_%s = sipModuleAPI_%s.em_imports[%d].im_module;\n" + ,mname,mld->module->name,mname,mld->module->modulenr); + } + + generateClassesInline(pt,fp); + generateEnumsInline(pt, fp); + + /* Create any exceptions. */ + for (xd = pt->exceptions; xd != NULL; xd = xd->next) + { + if (xd->iff->module != pt->module) + continue; + + if (xd->iff->type != exception_iface) + continue; + + if (xd->exceptionnr < 0) + continue; + + prcode(fp, +"\n" +" if ((exceptionsTable[%d] = PyErr_NewException((char *)\"%s.%s\",", xd->exceptionnr, xd->iff->module->name, xd->pyname); + + if (xd->bibase != NULL) + prcode(fp, "PyExc_%s", xd->bibase); + else if (xd->base->iff->module == pt->module) + prcode(fp, "exceptionsTable[%d]", xd->base->exceptionnr); + else + prcode(fp, "sipException_%C", xd->base->iff->fqcname); + + prcode(fp, ",NULL)) == NULL || PyDict_SetItemString(sipModuleDict,\"%s\",exceptionsTable[%d]) < 0)\n" +" return;\n" + , xd->pyname, xd->exceptionnr); + } + + /* Generate any Qt metatype registration calls. */ + if (optRegisterTypes(pt)) + for (cd = pt->classes; cd != NULL; cd = cd->next) + { + if (cd->iff->module != pt->module) + continue; + + generateRegisterMetaType(cd, fp); + } + + /* Generate the post-initialisation code. */ + generateCppCodeBlock(pt->postinitcode,fp); + + /* + * This has to be done after the post-initialisation code in case this + * module is exporting the symbol. + */ + if (optQ_OBJECT4(pt)) + prcode(fp, +"\n" +" sip_%s_qt_metaobject = (sip_qt_metaobject_func)sipImportSymbol(\"qtcore_qt_metaobject\");\n" +" sip_%s_qt_metacall = (sip_qt_metacall_func)sipImportSymbol(\"qtcore_qt_metacall\");\n" + , mname + , mname); + + prcode(fp, +"}\n" + ); + + /* Generate the interface source and header files. */ + for (iff = pt->ifacefiles; iff != NULL; iff = iff->next) + { + if (iff->module == pt->module && iff->type != exception_iface) + { + if (*parts && files_in_part++ == max_per_part) + { + /* Close the old part. */ + closeFile(fp); + free(cppfile); + + /* Create a new one. */ + files_in_part = 1; + ++this_part; + + cppfile = makePartName(codeDir,mname,this_part,srcSuffix); + fp = createCompilationUnit(pt, cppfile, "Module code."); + } + + generateIfaceCpp(pt,iff,codeDir,srcSuffix,(*parts ? fp : NULL)); + } + + generateIfaceHeader(pt,iff,codeDir); + } + + closeFile(fp); + free(cppfile); + + /* How many parts we actually generated. */ + if (*parts) + *parts = this_part + 1; +} + + +/* + * Generate all the sub-class convertors for a module. + */ +static int generateSubClassConvertors(sipSpec *pt, FILE *fp) +{ + int nrSccs = 0; + classDef *cd; + + for (cd = pt->classes; cd != NULL; cd = cd->next) + { + if (cd->iff->module != pt->module) + continue; + + if (cd->convtosubcode == NULL) + continue; + + prcode(fp, +"\n" +"\n" +"/* Convert to a sub-class if possible. */\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static sipWrapperType *sipSubClass_%C(void **);}\n" + , classFQCName(cd)); + + prcode(fp, +"static sipWrapperType *sipSubClass_%C(void **sipCppRet)\n" +"{\n" +" %S *sipCpp = reinterpret_cast<%S *>(*sipCppRet);\n" +" sipWrapperType *sipClass;\n" +"\n" + , classFQCName(cd) + , classFQCName(cd->subbase), classFQCName(cd->subbase)); + + generateCppCodeBlock(cd->convtosubcode, fp); + + prcode(fp, +"\n" +" return sipClass;\n" +"}\n" + ); + + ++nrSccs; + } + + return nrSccs; +} + + +/* + * Generate an entry for a class in the classes table and all its children. + */ +static void generateClassTableEntries(sipSpec *pt, nodeDef *nd, FILE *fp) +{ + nodeDef *cnd; + + /* Generate the entry if it's not the root. */ + if (nd->cd != NULL) + if (isExternal(nd->cd)) + prcode(fp, +" 0,\n" + ); + else + prcode(fp, +" (sipWrapperType *)(void *)&sipType_%s_%C,\n" + , pt->module->name, classFQCName(nd->cd)); + + /* Generate all it's children. */ + for (cnd = nd->child; cnd != NULL; cnd = cnd->next) + generateClassTableEntries(pt, cnd, fp); +} + + +/* + * Generate the structure representing an encoded class. + */ +static void generateEncodedClass(sipSpec *pt,classDef *cd,int last,FILE *fp) +{ + moduleDef *mod = cd->iff->module; + + prcode(fp,"{%u, ",cd->classnr); + + if (mod == pt->module) + prcode(fp,"255"); + else + prcode(fp,"%u",mod->modulenr); + + prcode(fp,", %u}",last); +} + + +/* + * Generate an ordinary function (ie. not a class method). + */ +static void generateOrdinaryFunction(sipSpec *pt,classDef *cd,memberDef *md, + FILE *fp) +{ + overDef *od; + + prcode(fp, +"\n" +"\n" + ); + + if (cd != NULL) + { + if (!generating_c) + prcode(fp, +"extern \"C\" {static PyObject *meth_%C_%s(PyObject *,PyObject *);}\n" + , classFQCName(cd), md->pyname->text); + + prcode(fp, +"static PyObject *meth_%C_%s(PyObject *,PyObject *sipArgs)\n" + ,classFQCName(cd),md->pyname->text); + + od = cd->overs; + } + else + { + if (!generating_c) + prcode(fp, +"extern \"C\" {static PyObject *func_%s(PyObject *,PyObject *);}\n" + , md->pyname->text); + + prcode(fp, +"static PyObject *func_%s(PyObject *%s,PyObject *sipArgs)\n" + ,md->pyname->text,(generating_c ? "sipSelf" : "")); + + od = pt->overs; + } + + prcode(fp, +"{\n" +" int sipArgsParsed = 0;\n" + ); + + while (od != NULL) + { + if (od->common == md) + generateFunctionBody(pt,od,cd,cd,TRUE,fp); + + od = od->next; + } + + prcode(fp, +"\n" +" /* Raise an exception if the arguments couldn't be parsed. */\n" +" sipNoFunction(sipArgsParsed,%N);\n" +"\n" +" return NULL;\n" +"}\n" + ,md->pyname); +} + + +/* + * Generate the table of enum members for a scope. Return the number of them. + */ +static int generateEnumMemberTable(sipSpec *pt,classDef *cd,FILE *fp) +{ + int i, nr_members; + enumDef *ed; + enumMemberDef **etab, **et; + + /* First we count how many. */ + + nr_members = 0; + + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + enumMemberDef *emd; + + if (ed->ecd != cd || ed->module != pt->module) + continue; + + if (cd == NULL && ed->fqcname == NULL) + continue; + + for (emd = ed->members; emd != NULL; emd = emd->next) + ++nr_members; + } + + if (nr_members == 0) + return 0; + + /* Create a table so they can be sorted. */ + + etab = sipMalloc(sizeof (enumMemberDef *) * nr_members); + + et = etab; + + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + enumMemberDef *emd; + + if (ed->ecd != cd || ed->module != pt->module) + continue; + + if (cd == NULL && ed->fqcname == NULL) + continue; + + for (emd = ed->members; emd != NULL; emd = emd->next) + *et++ = emd; + } + + qsort(etab,nr_members,sizeof (enumMemberDef *),compareEnumMembers); + + /* Now generate the table. */ + + if (cd != NULL) + prcode(fp, +"\n" +"static sipEnumMemberDef enummembers_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"/* These are the enum members of all global enums. */\n" +"static sipEnumMemberDef enummembers[] = {\n" + ); + + for (i = 0; i < nr_members; ++i) + { + enumMemberDef *emd; + + emd = etab[i]; + + prcode(fp, +" {%N, ",emd->pyname); + + if (cd != NULL) + { + if (isProtectedEnum(emd->ed)) + prcode(fp,"sip"); + + prcode(fp,"%S::%s",classFQCName(cd),emd->cname); + } + else + prcode(fp,"%s" + ,emd->cname); + + prcode(fp, ", %d},\n", emd->ed->enumnr); + } + + prcode(fp, +"};\n" + ); + + return nr_members; +} + + +/* + * The qsort helper to compare two enumMemberDef structures based on the name + * of the enum member. + */ + +static int compareEnumMembers(const void *m1,const void *m2) +{ + return strcmp((*(enumMemberDef **)m1)->pyname->text, + (*(enumMemberDef **)m2)->pyname->text); +} + + +/* + * Generate the access functions for the variables. + */ +static void generateAccessFunctions(sipSpec *pt,classDef *cd,FILE *fp) +{ + varDef *vd; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + if (vd->accessfunc == NULL) + continue; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + prcode(fp, +"\n" +"\n" +"/* Access function. */\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void *access_%C();}\n" + , vd->fqcname); + + prcode(fp, +"static void *access_%C()\n" +"{\n" + ,vd->fqcname); + + generateCppCodeBlock(vd->accessfunc,fp); + + prcode(fp, +"}\n" + ); + } +} + + +/* + * Generate the inline code to add a set of enum instances to a dictionary. + */ +static void generateEnumsInline(sipSpec *pt, FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + if (vd->module != pt->module) + continue; + + if (vd->type.atype != enum_type) + continue; + + if (needsHandler(vd)) + continue; + + /* Skip enums that don't need inline code. */ + if (generating_c || vd->accessfunc != NULL || vd->type.nrderefs != 0) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +" /* Define the enum instances that have to be added inline. */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" sipAddEnumInstance("); + + if (vd->ecd == NULL) + prcode(fp,"sipModuleDict"); + else + prcode(fp,"(PyObject *)sipClass_%C",classFQCName(vd->ecd)); + + prcode(fp,",%N,(int)%S,sipEnum_%C);\n" + , vd->pyname, vd->fqcname, vd->type.u.ed->fqcname); + } +} + + +/* + * Generate the inline code to add a set of class instances to a dictionary. + */ +static void generateClassesInline(sipSpec *pt,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + if (vd->module != pt->module) + continue; + + if (vd->type.atype != class_type && vd->type.atype != mapped_type) + continue; + + if (needsHandler(vd)) + continue; + + /* Skip classes that don't need inline code. */ + if (generating_c || vd->accessfunc != NULL || vd->type.nrderefs != 0) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +" /*\n" +" * Define the class and mapped type instances that have to be added\n" +" * inline.\n" +" */\n" + ); + + noIntro = FALSE; + } + + if (vd->type.atype == class_type) + prcode(fp, +" sipAddClassInstance("); + else + prcode(fp, +" sipAddMappedTypeInstance("); + + if (vd->ecd == NULL) + prcode(fp,"sipModuleDict"); + else + prcode(fp,"(PyObject *)sipClass_%C",classFQCName(vd->ecd)); + + prcode(fp,",%N,",vd->pyname); + + if (isConstArg(&vd->type)) + prcode(fp,"const_cast<%b *>(&%S)",&vd->type,vd->fqcname); + else + prcode(fp,"&%S",vd->fqcname); + + if (vd->type.atype == class_type) + prcode(fp, ",sipClass_%C);\n" + , classFQCName(vd->type.u.cd)); + else + prcode(fp, ",sipMappedType_%T);\n" + , &vd->type); + } +} + + +/* + * Generate the code to add a set of class instances to a dictionary. Return + * TRUE if there was at least one. + */ +static int generateClasses(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + scopedNameDef *vcname; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (vd->type.atype != class_type) + continue; + + if (needsHandler(vd)) + continue; + + /* + * Skip ordinary C++ class instances which need to be done with + * inline code rather than through a static table. This is + * because C++ does not guarantee the order in which the table + * and the instance will be created. So far this has only been + * seen to be a problem when statically linking SIP generated + * modules on Windows. + */ + if (!generating_c && vd->accessfunc == NULL && vd->type.nrderefs == 0) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the class instances to be added to this type dictionary. */\n" +"static sipClassInstanceDef classInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the class instances to be added to this module dictionary. */\n" +"static sipClassInstanceDef classInstances[] = {\n" + ); + + noIntro = FALSE; + } + + vcname = classFQCName(vd->type.u.cd); + + if (vd->accessfunc != NULL) + { + prcode(fp, +" {%N, (void *)access_%C, &sipClass_%C, SIP_ACCFUNC},\n" + ,vd->pyname,vd->fqcname,vcname); + } + else if (vd->type.nrderefs != 0) + { + prcode(fp, +" {%N, &%S, &sipClass_%C, SIP_INDIRECT},\n" + ,vd->pyname,vd->fqcname,vcname); + } + else if (isConstArg(&vd->type)) + { + prcode(fp, +" {%N, const_cast<%b *>(&%S), &sipClass_%C, 0},\n" + ,vd->pyname,&vd->type,vd->fqcname,vcname); + } + else + { + prcode(fp, +" {%N, &%S, &sipClass_%C, 0},\n" + ,vd->pyname,vd->fqcname,vcname); + } + } + + if (!noIntro) + prcode(fp, +" {0, 0, 0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of void pointers to a dictionary. Return + * TRUE if there was at least one. + */ +static int generateVoidPointers(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (vd->type.atype != void_type && vd->type.atype != struct_type) + continue; + + if (needsHandler(vd)) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the void pointers to be added to this type dictionary. */\n" +"static sipVoidPtrInstanceDef voidPtrInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the void pointers to be added to this module dictionary. */\n" +"static sipVoidPtrInstanceDef voidPtrInstances[] = {\n" + ); + + noIntro = FALSE; + } + + if (isConstArg(&vd->type)) + prcode(fp, +" {%N, const_cast<%b *>(%S)},\n" + , vd->pyname, &vd->type, vd->fqcname); + else + prcode(fp, +" {%N, %S},\n" + , vd->pyname, vd->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of characters to a dictionary. Return TRUE + * if there was at least one. + */ +static int generateChars(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + argType vtype = vd->type.atype; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (!((vtype == sstring_type || vtype == ustring_type || vtype == string_type || vtype == wstring_type) && vd->type.nrderefs == 0)) + continue; + + if (needsHandler(vd)) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the chars to be added to this type dictionary. */\n" +"static sipCharInstanceDef charInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the chars to be added to this module dictionary. */\n" +"static sipCharInstanceDef charInstances[] = {\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, %S},\n" + ,vd->pyname,vd->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of strings to a dictionary. Return TRUE if + * there is at least one. + */ +static int generateStrings(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + argType vtype = vd->type.atype; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (!((vtype == sstring_type || vtype == ustring_type || vtype == string_type || vtype == wstring_type) && vd->type.nrderefs != 0)) + continue; + + if (needsHandler(vd)) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the strings to be added to this type dictionary. */\n" +"static sipStringInstanceDef stringInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the strings to be added to this module dictionary. */\n" +"static sipStringInstanceDef stringInstances[] = {\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, %S},\n" + ,vd->pyname,vd->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of enum instances to a dictionary. Return + * TRUE if there was at least one. + */ +static int generateEnums(sipSpec *pt, classDef *cd, FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (vd->type.atype != enum_type || vd->type.u.ed->fqcname == NULL) + continue; + + if (needsHandler(vd)) + continue; + + /* Skip enums that need inline code. */ + if (!generating_c && vd->accessfunc == NULL && vd->type.nrderefs == 0) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the enum instances to be added to this type dictionary. */\n" +"static sipEnumInstanceDef enumInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the enum instances to be added to this module dictionary. */\n" +"static sipEnumInstanceDef enumInstances[] = {\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, (int)%S, &sipEnum_%C},\n" + ,vd->pyname,vd->fqcname,vd->type.u.ed->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of ints to a dictionary. Return TRUE if + * there was at least one. + */ +static int generateInts(sipSpec *pt, classDef *cd, FILE *fp) +{ + int noIntro; + varDef *vd; + enumDef *ed; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + argType vtype = vd->type.atype; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (!(vtype == enum_type || vtype == ushort_type || + vtype == short_type || vtype == uint_type || + vtype == cint_type || vtype == int_type || + vtype == bool_type || vtype == cbool_type)) + continue; + + if (needsHandler(vd)) + continue; + + /* Named enums are handled elsewhere. */ + if (vtype == enum_type && vd->type.u.ed->fqcname != NULL) + continue; + + if (noIntro) + { + ints_intro(cd, fp); + noIntro = FALSE; + } + + prcode(fp, +" {%N, %S},\n" + ,vd->pyname,vd->fqcname); + } + + /* Now do global anonymous enums. */ + if (cd == NULL) + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + enumMemberDef *em; + + if (ed->ecd != cd || ed->module != pt->module) + continue; + + if (ed->fqcname != NULL) + continue; + + for (em = ed->members; em != NULL; em = em->next) + { + if (noIntro) + { + ints_intro(cd, fp); + noIntro = FALSE; + } + + prcode(fp, +" {%N, %s},\n" + , em->pyname, em->cname); + } + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the intro for a table of int instances. + */ +static void ints_intro(classDef *cd, FILE *fp) +{ + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the ints to be added to this type dictionary. */\n" +"static sipIntInstanceDef intInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the ints to be added to this module dictionary. */\n" +"static sipIntInstanceDef intInstances[] = {\n" + ); +} + + +/* + * Generate the code to add a set of longs to a dictionary. Return TRUE if + * there was at least one. + */ +static int generateLongs(sipSpec *pt, classDef *cd, FILE *fp) +{ + return generateVariableType(pt, cd, long_type, "long", "Long", "long", fp); +} + + +/* + * Generate the code to add a set of unsigned longs to a dictionary. Return + * TRUE if there was at least one. + */ +static int generateUnsignedLongs(sipSpec *pt, classDef *cd, FILE *fp) +{ + return generateVariableType(pt, cd, ulong_type, "unsigned long", "UnsignedLong", "unsignedLong", fp); +} + + +/* + * Generate the code to add a set of long longs to a dictionary. Return TRUE + * if there was at least one. + */ +static int generateLongLongs(sipSpec *pt, classDef *cd, FILE *fp) +{ + return generateVariableType(pt, cd, longlong_type, "long long", "LongLong", "longLong", fp); +} + + +/* + * Generate the code to add a set of unsigned long longs to a dictionary. + * Return TRUE if there was at least one. + */ +static int generateUnsignedLongLongs(sipSpec *pt, classDef *cd, FILE *fp) +{ + return generateVariableType(pt, cd, ulonglong_type, "unsigned long long", "UnsignedLongLong", "unsignedLongLong", fp); +} + + +/* + * Generate the code to add a set of a particular type to a dictionary. Return + * TRUE if there was at least one. + */ +static int generateVariableType(sipSpec *pt, classDef *cd, argType atype, const char *eng, const char *s1, const char *s2, FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + argType vtype = vd->type.atype; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (vtype != atype) + continue; + + if (needsHandler(vd)) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the %ss to be added to this type dictionary. */\n" +"static sip%sInstanceDef %sInstances_%C[] = {\n" + , eng + , s1, s2, classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the %ss to be added to this module dictionary. */\n" +"static sip%sInstanceDef %sInstances[] = {\n" + , eng + , s1, s2); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, %S},\n" + ,vd->pyname,vd->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the code to add a set of doubles to a dictionary. Return TRUE if + * there was at least one. + */ +static int generateDoubles(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro; + varDef *vd; + + noIntro = TRUE; + + for (vd = pt->vars; vd != NULL; vd = vd->next) + { + argType vtype = vd->type.atype; + + if (vd->ecd != cd || vd->module != pt->module) + continue; + + if (!(vtype == float_type || vtype == cfloat_type || vtype == double_type || vtype == cdouble_type)) + continue; + + if (needsHandler(vd)) + continue; + + if (noIntro) + { + if (cd != NULL) + prcode(fp, +"\n" +"\n" +"/* Define the doubles to be added to this type dictionary. */\n" +"static sipDoubleInstanceDef doubleInstances_%C[] = {\n" + ,classFQCName(cd)); + else + prcode(fp, +"\n" +"\n" +"/* Define the doubles to be added to this module dictionary. */\n" +"static sipDoubleInstanceDef doubleInstances[] = {\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, %S},\n" + ,vd->pyname,vd->fqcname); + } + + if (!noIntro) + prcode(fp, +" {0, 0}\n" +"};\n" + ); + + return !noIntro; +} + + +/* + * Generate the C/C++ code for an interface. + */ +static void generateIfaceCpp(sipSpec *pt,ifaceFileDef *iff,char *codeDir, + char *srcSuffix,FILE *master) +{ + char *cppfile, *cmname = iff->module->name; + classDef *cd; + mappedTypeDef *mtd; + FILE *fp; + + if (master == NULL) + { + cppfile = createIfaceFileName(codeDir,iff,srcSuffix); + fp = createCompilationUnit(pt, cppfile, "Interface wrapper code."); + } + else + fp = master; + + prcode(fp, +"\n" +"#include \"sipAPI%s.h\"\n" +"#include \"sip%s%F.h\"\n" + ,cmname + ,cmname,iff->fqcname); + + generateUsedIncludes(iff->used, FALSE, fp); + + for (cd = pt->classes; cd != NULL; cd = cd->next) + if (cd->iff == iff) + { + if (isProtectedClass(cd)) + prcode(fp, +"\n" +"#include \"sip%s%F.h\"\n" + ,cmname,cd->ecd->iff->fqcname); + + if (!isExternal(cd)) + generateClassCpp(cd, pt, fp); + } + + for (mtd = pt->mappedtypes; mtd != NULL; mtd = mtd->next) + if (mtd->iff == iff) + generateMappedTypeCpp(mtd,fp); + + if (master == NULL) + { + closeFile(fp); + free(cppfile); + } +} + + +/* + * Return a filename for an interface C++ or header file on the heap. + */ +static char *createIfaceFileName(char *codeDir,ifaceFileDef *iff,char *suffix) +{ + char *fn; + scopedNameDef *snd; + + fn = concat(codeDir,"/sip",iff->module->name,NULL); + + for (snd = iff->fqcname; snd != NULL; snd = snd->next) + append(&fn,snd->name); + + append(&fn,suffix); + + return fn; +} + + +/* + * Generate the C++ code for a mapped type version. + */ +static void generateMappedTypeCpp(mappedTypeDef *mtd,FILE *fp) +{ + int need_xfer; + + prcode(fp, +"\n" +"\n" +"/* Call the mapped type's destructor. */\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void release_%T(void *, int);}\n" + , &mtd->type); + + prcode(fp, +"static void release_%T(void *ptr, int%s)\n" +"{\n" + , &mtd->type, (generating_c ? " status" : "")); + + if (release_gil) + prcode(fp, +" Py_BEGIN_ALLOW_THREADS\n" + ); + + if (generating_c) + prcode(fp, +" sipFree(ptr);\n" + ); + else + prcode(fp, +" delete reinterpret_cast<%b *>(ptr);\n" + , &mtd->type); + + if (release_gil) + prcode(fp, +" Py_END_ALLOW_THREADS\n" + ); + + prcode(fp, +"}\n" +"\n" + ); + + generateConvertToDefinitions(mtd,NULL,fp); + + /* Generate the from type convertor. */ + + need_xfer = (generating_c || usedInCode(mtd->convfromcode, "sipTransferObj")); + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static PyObject *convertFrom_%T(void *, PyObject *);}\n" + , &mtd->type); + + prcode(fp, +"static PyObject *convertFrom_%T(void *sipCppV,PyObject *%s)\n" +"{\n" +" ", &mtd->type, (need_xfer ? "sipTransferObj" : "")); + + generateMappedTypeFromVoid(mtd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +"\n" + ); + + generateCppCodeBlock(mtd->convfromcode,fp); + + prcode(fp, +"}\n" +"\n" +"\n" +"sipMappedType sipMappedTypeDef_%T = {\n" +" \"%B\",\n" +" release_%T,\n" +" forceConvertTo_%T,\n" +" convertTo_%T,\n" +" convertFrom_%T\n" +"};\n" + , &mtd->type + , &mtd->type + , &mtd->type + , &mtd->type + , &mtd->type + , &mtd->type + , &mtd->type); +} + + +/* + * Generate the C++ code for a class. + */ +static void generateClassCpp(classDef *cd,sipSpec *pt,FILE *fp) +{ + varDef *vd; + + /* Generate any local class code. */ + + generateCppCodeBlock(cd->cppcode,fp); + + generateClassFunctions(pt,cd,fp); + + generateAccessFunctions(pt,cd,fp); + + /* Generate the variable handlers. */ + if (hasVarHandlers(cd)) + { + for (vd = pt->vars; vd != NULL; vd = vd->next) + if (vd->ecd == cd && needsHandler(vd)) + generateVariableHandler(vd,fp); + + /* Generate the variable table. */ + prcode(fp, +"\n" +"PyMethodDef variables_%C[] = {\n" + ,classFQCName(cd)); + + for (vd = pt->vars; vd != NULL; vd = vd->next) + if (vd->ecd == cd && needsHandler(vd)) + prcode(fp, +" {%N, var_%C, %s, NULL},\n" + ,vd->pyname,vd->fqcname,(isStaticVar(vd) ? "METH_STATIC" : "0")); + + prcode(fp, +" {0, 0, 0, 0}\n" +"};\n" + ); + } + + if (cd->iff->type != namespace_iface) + generateConvertToDefinitions(NULL,cd,fp); + + /* The type definition structure. */ + generateTypeDefinition(pt, cd, fp); +} + + +/* + * Return a sorted array of relevant functions for a namespace. + */ + +static sortedMethTab *createFunctionTable(classDef *cd,int *nrp) +{ + int nr; + sortedMethTab *mtab, *mt; + memberDef *md; + + /* First we need to count the number of applicable functions. */ + + nr = 0; + + for (md = cd->members; md != NULL; md = md->next) + ++nr; + + if ((*nrp = nr) == 0) + return NULL; + + /* Create the table of methods. */ + + mtab = sipMalloc(sizeof (sortedMethTab) * nr); + + /* Initialise the table. */ + + mt = mtab; + + for (md = cd->members; md != NULL; md = md->next) + { + mt->md = md; + mt->is_static = TRUE; + + ++mt; + } + + /* Finally sort the table. */ + + qsort(mtab,nr,sizeof (sortedMethTab),compareMethTab); + + return mtab; +} + + +/* + * Return a sorted array of relevant methods (either lazy or non-lazy) for a + * class. + */ +static sortedMethTab *createMethodTable(classDef *cd, int *nrp) +{ + int nr; + visibleList *vl; + sortedMethTab *mtab, *mt; + + /* + * First we need to count the number of applicable methods. Only provide + * an entry point if there is at least one overload that is defined in this + * class and is a non-abstract function or slot. We allow private (even + * though we don't actually generate code) because we need to intercept the + * name before it reaches a more public version further up the class + * hierarchy. We add the ctor and any variable handlers as special + * entries. + */ + nr = 0; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + if (vl->m->slot != no_slot) + continue; + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + /* + * Skip protected methods if we don't have the means to + * handle them. + */ + if (isProtected(od) && !hasShadow(cd)) + continue; + + if (skipOverload(od,vl->m,cd,vl->cd,TRUE)) + continue; + + ++nr; + + break; + } + } + + if ((*nrp = nr) == 0) + return NULL; + + /* Create the table of methods. */ + + mtab = sipMalloc(sizeof (sortedMethTab) * nr); + + /* Initialise the table. */ + + mt = mtab; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + int need_method, is_static; + overDef *od; + + if (vl->m->slot != no_slot) + continue; + + need_method = FALSE; + is_static = TRUE; + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + /* + * Skip protected methods if we don't have the means to + * handle them. + */ + if (isProtected(od) && !hasShadow(cd)) + continue; + + if (!skipOverload(od,vl->m,cd,vl->cd,TRUE)) + { + need_method = TRUE; + + if (!isPrivate(od) && !isStatic(od)) + is_static = FALSE; + } + } + + if (need_method) + { + mt->md = vl->m; + mt->is_static = is_static; + + ++mt; + } + } + + /* Finally sort the table. */ + + qsort(mtab,nr,sizeof (sortedMethTab),compareMethTab); + + return mtab; +} + + +/* + * The qsort helper to compare two sortedMethTab structures based on the Python + * name of the method. + */ + +static int compareMethTab(const void *m1,const void *m2) +{ + return strcmp(((sortedMethTab *)m1)->md->pyname->text, + ((sortedMethTab *)m2)->md->pyname->text); +} + + +/* + * Generate the sorted table of methods and return the number of entries. + */ +static int generateMethodTable(classDef *cd,FILE *fp) +{ + int nr; + sortedMethTab *mtab; + + mtab = (cd->iff->type == namespace_iface) ? + createFunctionTable(cd,&nr) : + createMethodTable(cd,&nr); + + if (mtab != NULL) + { + int i; + + prcode(fp, +"\n" +"\n" +"static PyMethodDef methods_%C[] = {\n" + ,classFQCName(cd)); + + for (i = 0; i < nr; ++i) + { + memberDef *md = mtab[i].md; + + /* + * For the moment we are suppressing the generation of + * METH_STATIC until we understand descriptors better. + * It could be that they will simplify the handling of + * lazy attributes and allow things to be cached in the + * type dictionary. + */ + mtab[i].is_static = FALSE; + + prcode(fp, +" {%N, meth_%C_%s, METH_VARARGS%s, NULL}%s\n" + ,md->pyname,classFQCName(cd),md->pyname->text,(mtab[i].is_static ? "|METH_STATIC" : ""),((i + 1) < nr) ? "," : ""); + } + + free(mtab); + + prcode(fp, +"};\n" + ); + } + + return nr; +} + + +/* + * Generate the "to type" convertor definitions. + */ + +static void generateConvertToDefinitions(mappedTypeDef *mtd,classDef *cd, + FILE *fp) +{ + codeBlock *convtocode; + ifaceFileDef *iff; + argDef type; + + if (cd != NULL) + { + convtocode = cd->convtocode; + + iff = cd->iff; + + type.atype = class_type; + type.u.cd = cd; + } + else + { + convtocode = mtd->convtocode; + + iff = mtd->iff; + + type.atype = mapped_type; + type.u.mtd = mtd; + } + + type.argflags = 0; + type.name = NULL; + type.nrderefs = 0; + type.defval = NULL; + + /* Generate the type convertors. */ + + if (convtocode != NULL) + { + int need_ptr, need_xfer; + + /* + * Sometimes type convertors are just stubs that set the error + * flag, so check if we actually need everything so that we + * can avoid compiler warnings. + */ + need_ptr = (generating_c || usedInCode(convtocode, "sipCppPtr")); + need_xfer = (generating_c || usedInCode(convtocode, "sipTransferObj")); + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static int convertTo_%T(PyObject *, void **, int *, PyObject *);}\n" + , &type); + + prcode(fp, +"static int convertTo_%T(PyObject *sipPy,void **%s,int *sipIsErr,PyObject *%s)\n" +"{\n" + , &type, (need_ptr ? "sipCppPtrV" : ""), (need_xfer ? "sipTransferObj" : "")); + + if (need_ptr) + if (generating_c) + prcode(fp, +" %b **sipCppPtr = (%b **)sipCppPtrV;\n" +"\n" + , &type, &type); + else + prcode(fp, +" %b **sipCppPtr = reinterpret_cast<%b **>(sipCppPtrV);\n" +"\n" + , &type, &type); + + generateCppCodeBlock(convtocode,fp); + + prcode(fp, +"}\n" + ); + } + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void *forceConvertTo_%T(PyObject *, int *);}\n" + , &type); + + prcode(fp, +"static void *forceConvertTo_%T(PyObject *valobj,int *iserrp)\n" +"{\n" +" if (*iserrp || valobj == NULL)\n" +" return NULL;\n" +"\n" + ,&type); + + if (convtocode != NULL) + prcode(fp, +" if (convertTo_%T(valobj,NULL,NULL,NULL))\n" +" {\n" +" void *val;\n" +"\n" +" /*\n" +" * Note that we throw away the flag that says if the value\n" +" * has just been created on the heap or not.\n" +" */\n" +" convertTo_%T(valobj,&val,iserrp,NULL);\n" +"\n" +" return val;\n" +" }\n" + ,&type + ,&type); + else + prcode(fp, +" if (valobj == Py_None || sipIsSubClassInstance(valobj,sipClass_%T))\n" +" return sipConvertToCpp(valobj,sipClass_%T,iserrp);\n" + ,&type + ,&type); + + if (cd != NULL) + prcode(fp, +"\n" +" sipBadClass(%N);\n" + , iff->name); + else + prcode(fp, +"\n" +" sipBadClass(\"%B\");\n" + , &mtd->type); + + prcode(fp, +"\n" +" *iserrp = 1;\n" +"\n" +" return NULL;\n" +"}\n" + ); +} + + +/* + * Generate a variable handler. + */ +static void generateVariableHandler(varDef *vd,FILE *fp) +{ + argType atype = vd->type.atype; + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static PyObject *var_%C(PyObject *, PyObject *);}\n" + , vd->fqcname); + + prcode(fp, +"static PyObject *var_%C(PyObject *%s,PyObject *sipPy)\n" +"{\n" + ,vd->fqcname,(isStaticVar(vd) ? "" : "sipSelf")); + + if (atype == class_type || atype == mapped_type) + prcode(fp, +" int sipIsErr = 0;\n" + ); + + if (vd->type.nrderefs == 0 && (atype == mapped_type || (atype == class_type && vd->type.u.cd->convtocode != NULL))) + prcode(fp, +" int sipValState;\n" + ); + + if (vd->getcode == NULL || vd->setcode == NULL) + { + prcode(fp, +" "); + + generateNamedValueType(&vd->type, "sipVal", fp); + + prcode(fp, ";\n" + ); + } + + if (!isStaticVar(vd)) + { + if (generating_c) + prcode(fp, +" %S *sipCpp = (%S *)sipGetCppPtr((sipWrapper *)sipSelf,sipClass_%C);\n" + ,classFQCName(vd->ecd),classFQCName(vd->ecd),classFQCName(vd->ecd)); + else + prcode(fp, +" %S *sipCpp = reinterpret_cast<%S *>(sipGetCppPtr((sipWrapper *)sipSelf,sipClass_%C));\n" + ,classFQCName(vd->ecd),classFQCName(vd->ecd),classFQCName(vd->ecd)); + + prcode(fp, +"\n" +" if (!sipCpp)\n" +" return NULL;\n" + ); + } + + prcode(fp, +"\n" +" if (sipPy == NULL)\n" +" {\n" + ); + + /* Generate the get handler part. */ + + if (vd->getcode != NULL) + { + generateCppCodeBlock(vd->getcode, fp); + + prcode(fp, +"\n" +" return sipPy;\n" + ); + } + else + { + int pyobj = FALSE; + + prcode(fp, +" sipVal = %s", (((atype == class_type || atype == mapped_type) && vd->type.nrderefs == 0) ? "&" : "")); + + generateVarMember(vd, fp); + + prcode(fp, ";\n" +"\n" + ); + + switch (atype) + { + case mapped_type: + prcode(fp, +" sipPy = sipConvertFromMappedType(sipVal,sipMappedType_%T,NULL);\n" + ,&vd->type); + + break; + + case class_type: + generateVarClassConversion(vd,fp); + break; + + case bool_type: + case cbool_type: + prcode(fp, +" sipPy = PyBool_FromLong(sipVal);\n" + ); + + break; + + case sstring_type: + case ustring_type: + case string_type: + if (vd->type.nrderefs == 0) + prcode(fp, +" sipPy = PyString_FromStringAndSize(%s&sipVal,1);\n" + ,(atype != string_type) ? "(char *)" : ""); + else + prcode(fp, +" sipPy = PyString_FromString(%ssipVal);\n" + ,(atype != string_type) ? "(char *)" : ""); + + break; + + case wstring_type: + if (vd->type.nrderefs == 0) + prcode(fp, +" sipPy = PyUnicode_FromWideChar(&sipVal,1);\n" + ); + else + prcode(fp, +" sipPy = PyUnicode_FromWideChar(sipVal,(SIP_SSIZE_T)wcslen(sipVal));\n" + ); + + break; + + case float_type: + case cfloat_type: + prcode(fp, +" sipPy = PyFloat_FromDouble((double)sipVal);\n" + ); + break; + + case double_type: + case cdouble_type: + prcode(fp, +" sipPy = PyFloat_FromDouble(sipVal);\n" + ); + break; + + case enum_type: + if (vd->type.u.ed->fqcname != NULL) + { + prcode(fp, +" sipPy = sipConvertFromNamedEnum(sipVal,sipEnum_%C);\n" + ,vd->type.u.ed->fqcname); + + break; + } + + /* Drop through. */ + + case short_type: + case cint_type: + case int_type: + prcode(fp, +" sipPy = PyInt_FromLong(sipVal);\n" + ); + break; + + case long_type: + prcode(fp, +" sipPy = PyLong_FromLong(sipVal);\n" + ); + break; + + case ushort_type: + case uint_type: + case ulong_type: + prcode(fp, +" sipPy = PyLong_FromUnsignedLong(sipVal);\n" + ); + break; + + case longlong_type: + prcode(fp, +" sipPy = PyLong_FromLongLong(sipVal);\n" + ); + break; + + case ulonglong_type: + prcode(fp, +" sipPy = PyLong_FromUnsignedLongLong(sipVal);\n" + ); + break; + + case struct_type: + prcode(fp, +" sipPy = sipConvertFromVoidPtr("); + + if (isConstArg(&vd->type)) + prcode(fp, "const_cast<%b *>(sipVal)", &vd->type); + else + prcode(fp, "sipVal"); + + prcode(fp, ");\n" + ); + + break; + + case void_type: + prcode(fp, +" sipPy = sipConvertFromVoidPtr("); + + if (isConstArg(&vd->type)) + prcode(fp, "const_cast<void *>(sipVal)"); + else + prcode(fp, "sipVal"); + + prcode(fp, ");\n" + ); + + break; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + prcode(fp, +" Py_XINCREF(sipVal);\n" + ); + pyobj = TRUE; + break; + } + + prcode(fp, +"\n" +" return %s;\n" + ,(pyobj ? "sipVal" : "sipPy")); + } + + prcode(fp, +" }\n" +"\n" + ); + + /* Generate the set handler part. */ + + if (vd->setcode != NULL) + { + prcode(fp, +" {\n" +" int sipErr = 0;\n" +"\n" + ); + + generateCppCodeBlock(vd->setcode, fp); + + prcode(fp, +"\n" +" if (sipErr)\n" +" return NULL;\n" +" }\n" + ); + } + else + { + char *deref; + int might_be_temp; + + might_be_temp = generateObjToCppConversion(&vd->type,fp); + + deref = ""; + + if (atype == class_type || atype == mapped_type) + { + if (vd->type.nrderefs == 0) + deref = "*"; + + prcode(fp, +"\n" +" if (sipIsErr)\n" +" return NULL;\n" +"\n" + ); + } + else + { + if ((atype == sstring_type || atype == ustring_type || atype == string_type || atype == wstring_type) && vd->type.nrderefs != 0) + { + prcode(fp, +"\n" +" if (sipVal == NULL)\n" + ); + } + else + prcode(fp, +"\n" +" if (PyErr_Occurred() != NULL)\n" + ); + + prcode(fp, +" {\n" +" sipBadSetType(%N,%N);\n" +" return NULL;\n" +" }\n" +"\n" + ,vd->ecd->iff->name,vd->pyname); + } + + if (atype == pyobject_type || atype == pytuple_type || + atype == pylist_type || atype == pydict_type || + atype == pycallable_type || atype == pyslice_type || + atype == pytype_type) + { + prcode(fp, +" Py_XDECREF("); + + generateVarMember(vd, fp); + + prcode(fp, ");\n" +" Py_INCREF(sipVal);\n" +"\n" + ); + } + + prcode(fp, +" "); + + generateVarMember(vd, fp); + + prcode(fp, " = %ssipVal;\n" + , deref); + + /* Note that wchar_t * leaks here. */ + + if (might_be_temp) + prcode(fp, +"\n" +" sipReleaseInstance(sipVal,sipClass_%C,sipValState);\n" + , classFQCName(vd->type.u.cd)); + else if (vd->type.atype == mapped_type && vd->type.nrderefs == 0) + prcode(fp, +"\n" +" sipReleaseMappedType(sipVal,sipMappedType_%T,sipValState);\n" + , &vd->type); + } + + prcode(fp, +"\n" +" Py_INCREF(Py_None);\n" +" return Py_None;\n" +"}\n" + ); +} + + +/* + * Generate the member variable of a class. + */ +static void generateVarMember(varDef *vd, FILE *fp) +{ + if (isStaticVar(vd)) + prcode(fp,"%S::",classFQCName(vd->ecd)); + else + prcode(fp,"sipCpp->"); + + prcode(fp, "%s", scopedNameTail(vd->fqcname)); +} + + +/* + * Generate an variable class conversion fragment. + */ +static void generateVarClassConversion(varDef *vd,FILE *fp) +{ + classDef *cd = vd->type.u.cd; + + prcode(fp, +" sipPy = sipConvertFromInstance("); + + if (isConstArg(&vd->type)) + prcode(fp,"const_cast<%b *>(sipVal)",&vd->type); + else + prcode(fp,"sipVal"); + + prcode(fp,",sipClass_%C,NULL);\n" + ,classFQCName(cd)); +} + + +/* + * Generate the declaration of a variable that is initialised from a Python + * object. Return TRUE if the value might be a temporary on the heap. + */ +static int generateObjToCppConversion(argDef *ad,FILE *fp) +{ + int might_be_temp = FALSE; + char *rhs = NULL; + + prcode(fp, +" sipVal = "); + + switch (ad->atype) + { + case mapped_type: + { + const char *tail; + + if (generating_c) + { + prcode(fp, "(%b *)", ad); + tail = ""; + } + else + { + prcode(fp, "reinterpret_cast<%b *>(", ad); + tail = ")"; + } + + /* + * Note that we don't support /Transfer/ but could do. + */ + + prcode(fp, "sipForceConvertToMappedType(sipPy,sipMappedType_%T,NULL,%s,%s,&sipIsErr)", ad, (ad->nrderefs ? "0" : "SIP_NOT_NONE"), (ad->nrderefs ? "NULL" : "&sipValState")); + + prcode(fp, "%s;\n" + , tail); + } + break; + + case class_type: + { + const char *tail; + + if (ad->nrderefs == 0 && ad->u.cd->convtocode != NULL) + might_be_temp = TRUE; + + if (generating_c) + { + prcode(fp, "(%b *)", ad); + tail = ""; + } + else + { + prcode(fp, "reinterpret_cast<%b *>(", ad); + tail = ")"; + } + + /* + * Note that we don't support /Transfer/ but could do. + * We could also support /Constrained/ (so long as we + * also supported it for all types). + */ + + prcode(fp, "sipForceConvertToInstance(sipPy,sipClass_%C,NULL,%s,%s,&sipIsErr)", classFQCName(ad->u.cd), (ad->nrderefs ? "0" : "SIP_NOT_NONE"), (might_be_temp ? "&sipValState" : "NULL")); + + prcode(fp, "%s;\n" + , tail); + } + break; + + case enum_type: + prcode(fp, "(%E)PyInt_AsLong(sipPy);\n" + , ad->u.ed); + break; + + case sstring_type: + if (ad->nrderefs == 0) + rhs = "(signed char)sipString_AsChar(sipPy)"; + else + rhs = "(signed char *)PyString_AsString(sipPy)"; + break; + + case ustring_type: + if (ad->nrderefs == 0) + rhs = "(unsigned char)sipString_AsChar(sipPy)"; + else + rhs = "(unsigned char *)PyString_AsString(sipPy)"; + break; + + case string_type: + if (ad->nrderefs == 0) + rhs = "sipString_AsChar(sipPy)"; + else + rhs = "PyString_AsString(sipPy)"; + break; + + case wstring_type: + if (ad->nrderefs == 0) + rhs = "sipUnicode_AsWChar(sipPy)"; + else + rhs = "sipUnicode_AsWString(sipPy)"; + break; + + case float_type: + case cfloat_type: + rhs = "(float)PyFloat_AsDouble(sipPy)"; + break; + + case double_type: + case cdouble_type: + rhs = "PyFloat_AsDouble(sipPy)"; + break; + + case bool_type: + case cbool_type: + rhs = "(bool)PyInt_AsLong(sipPy)"; + break; + + case ushort_type: + rhs = "(unsigned short)sipLong_AsUnsignedLong(sipPy)"; + break; + + case short_type: + rhs = "(short)PyInt_AsLong(sipPy)"; + break; + + case uint_type: + rhs = "(unsigned)sipLong_AsUnsignedLong(sipPy)"; + break; + + case int_type: + case cint_type: + rhs = "(int)PyInt_AsLong(sipPy)"; + break; + + case ulong_type: + rhs = "sipLong_AsUnsignedLong(sipPy)"; + break; + + case long_type: + rhs = "PyLong_AsLong(sipPy)"; + break; + + case ulonglong_type: + rhs = "PyLong_AsUnsignedLongLong(sipPy)"; + break; + + case longlong_type: + rhs = "PyLong_AsLongLong(sipPy)"; + break; + + case struct_type: + prcode(fp, "(struct %S *)sipConvertToVoidPtr(sipPy);\n" + , ad->u.sname); + break; + + case void_type: + rhs = "sipConvertToVoidPtr(sipPy)"; + break; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + rhs = "sipPy"; + break; + } + + if (rhs != NULL) + prcode(fp, "%s;\n" + , rhs); + + return might_be_temp; +} + + +/* + * Returns TRUE if the given method is a slot that takes zero arguments. + */ +static int isZeroArgSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == str_slot || st == int_slot || st == long_slot || + st == float_slot || st == invert_slot || st == neg_slot || + st == len_slot || st == nonzero_slot || st == pos_slot || + st == abs_slot || st == repr_slot || st == hash_slot); +} + + +/* + * Returns TRUE if the given method is a slot that takes more than one + * argument. + */ +static int isMultiArgSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == setitem_slot || st == call_slot); +} + + +/* + * Returns TRUE if the given method is a slot that returns void (ie. nothing + * other than an error indicator). + */ +int isVoidReturnSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == setitem_slot || st == delitem_slot); +} + + +/* + * Returns TRUE if the given method is a slot that returns int. + */ +int isIntReturnSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == len_slot || st == nonzero_slot || st == contains_slot || + st == cmp_slot); +} + + +/* + * Returns TRUE if the given method is a slot that returns long. + */ +int isLongReturnSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == hash_slot); +} + + +/* + * Returns TRUE if the given method is a slot that takes an int argument. + */ +static int isIntArgSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == repeat_slot || st == irepeat_slot); +} + + +/* + * Returns TRUE if the given method is an inplace number slot. + */ +static int isInplaceNumberSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == iadd_slot || st == isub_slot || st == imul_slot || + st == idiv_slot || st == imod_slot || + st == ior_slot || st == ixor_slot || st == iand_slot || + st == ilshift_slot || st == irshift_slot); +} + + +/* + * Returns TRUE if the given method is an inplace sequence slot. + */ +static int isInplaceSequenceSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == iconcat_slot || st == irepeat_slot); +} + + +/* + * Returns TRUE if the given method is a number slot slot. + */ +int isNumberSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == add_slot || st == sub_slot || st == mul_slot || + st == div_slot || st == mod_slot || + st == and_slot || st == or_slot || st == xor_slot || + st == lshift_slot || st == rshift_slot); +} + + +/* + * Returns TRUE if the given method is a rich compare slot. + */ +int isRichCompareSlot(memberDef *md) +{ + slotType st = md->slot; + + return (st == lt_slot || st == le_slot || st == eq_slot || + st == ne_slot || st == gt_slot || st == ge_slot); +} + + +/* + * Generate a Python slot handler for either a class, an enum or an extender. + */ +static void generateSlot(sipSpec *pt, classDef *cd, enumDef *ed, memberDef *md, FILE *fp) +{ + char *arg_str, *prefix, *ret_type; + int ret_int, nr_args; + overDef *od, *overs; + scopedNameDef *fqcname; + nameDef *pyname; + + if (ed != NULL) + { + prefix = "Enum"; + pyname = ed->pyname; + fqcname = ed->fqcname; + overs = ed->overs; + } + else if (cd != NULL) + { + prefix = "Class"; + pyname = cd->iff->name; + fqcname = classFQCName(cd); + overs = cd->overs; + } + else + { + prefix = NULL; + pyname = NULL; + fqcname = NULL; + overs = pt->overs; + } + + if (isVoidReturnSlot(md) || isIntReturnSlot(md)) + { + ret_int = TRUE; + ret_type = "int "; + } + else + { + ret_int = FALSE; + + if (isLongReturnSlot(md)) + ret_type = "long "; + else + ret_type = "PyObject *"; + } + + if (isIntArgSlot(md)) + { + nr_args = 0; + arg_str = "PyObject *sipSelf,int a0"; + } + else if (isMultiArgSlot(md)) + { + nr_args = 2; + arg_str = "PyObject *sipSelf,PyObject *sipArgs"; + } + else if (isZeroArgSlot(md)) + { + nr_args = 0; + arg_str = "PyObject *sipSelf"; + } + else if (isNumberSlot(md)) + { + nr_args = 2; + arg_str = "PyObject *sipArg0,PyObject *sipArg1"; + } + else + { + nr_args = 1; + arg_str = "PyObject *sipSelf,PyObject *sipArg"; + } + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + { + prcode(fp, +"extern \"C\" {static %sslot_", ret_type); + + if (fqcname != NULL) + prcode(fp, "%C_", fqcname); + + prcode(fp, "%s(%s);}\n" + , md->pyname->text, arg_str); + } + + prcode(fp, +"static %sslot_", ret_type); + + if (fqcname != NULL) + prcode(fp, "%C_", fqcname); + + prcode(fp, "%s(%s)\n" +"{\n" + , md->pyname->text, arg_str); + + if (isInplaceNumberSlot(md)) + prcode(fp, +" if (!PyObject_TypeCheck(sipSelf,(PyTypeObject *)sip%s_%C))\n" +" {\n" +" Py_INCREF(Py_NotImplemented);\n" +" return Py_NotImplemented;\n" +" }\n" +"\n" + , prefix, fqcname); + + if (!isNumberSlot(md)) + if (cd != NULL) + prcode(fp, +" %S *sipCpp = reinterpret_cast<%S *>(sipGetCppPtr((sipWrapper *)sipSelf,sipClass_%C));\n" +"\n" +" if (!sipCpp)\n" +" return %s;\n" +"\n" + , fqcname, fqcname, fqcname + , (md->slot == cmp_slot ? "-2" : (ret_int ? "-1" : "0"))); + else + prcode(fp, +" %S sipCpp = static_cast<%S>(PyInt_AsLong(sipSelf));\n" +"\n" + , fqcname, fqcname); + + if (nr_args > 0) + prcode(fp, +" int sipArgsParsed = 0;\n" + ); + + for (od = overs; od != NULL; od = od->next) + if (od->common == md && isAbstract(od)) + { + prcode(fp, +" bool sipSelfWasArg = !sipSelf;\n" + ); + + break; + } + + for (od = overs; od != NULL; od = od->next) + if (od->common == md) + generateFunctionBody(pt, od, cd, cd, (ed == NULL && !dontDerefSelf(od)), fp); + + if (nr_args > 0) + switch (md->slot) + { + case cmp_slot: + prcode(fp, +"\n" +" return 2;\n" + ); + break; + + case concat_slot: + case iconcat_slot: + case repeat_slot: + case irepeat_slot: + prcode(fp, +"\n" +" /* Raise an exception if the argument couldn't be parsed. */\n" +" sipBadOperatorArg(sipSelf,sipArg,%s);\n" +"\n" +" return NULL;\n" + ,slotName(md->slot)); + break; + + default: + if (isNumberSlot(md) || isRichCompareSlot(md)) + { + /* We can't extend enum slots. */ + if (cd == NULL) + prcode(fp, +"\n" +" Py_INCREF(Py_NotImplemented);\n" +" return Py_NotImplemented;\n" + ); + else if (isNumberSlot(md)) + prcode(fp, +"\n" +" return sipPySlotExtend(&sipModuleAPI_%s,%s,NULL,sipArg0,sipArg1);\n" + , pt->module->name, slotName(md->slot)); + else + prcode(fp, +"\n" +" return sipPySlotExtend(&sipModuleAPI_%s,%s,sip%s_%C,sipSelf,sipArg);\n" + , pt->module->name, slotName(md->slot), prefix, fqcname); + } + else if (isInplaceNumberSlot(md)) + prcode(fp, +"\n" +" PyErr_Clear();\n" +"\n" +" Py_INCREF(Py_NotImplemented);\n" +" return Py_NotImplemented;\n" + ); + else + prcode(fp, +"\n" +" /* Raise an exception if the arguments couldn't be parsed. */\n" +" sipNoMethod(sipArgsParsed,%N,%N);\n" +"\n" +" return %s;\n" + , pyname, md->pyname + ,ret_int ? "-1" : "0"); + } + + prcode(fp, +"}\n" + ); +} + + +/* + * Generate the member functions for a class. + */ +static void generateClassFunctions(sipSpec *pt,classDef *cd,FILE *fp) +{ + visibleList *vl; + memberDef *md; + + /* Any shadow code. */ + if (hasShadow(cd)) + generateShadowCode(pt,cd,fp); + + /* The member functions. */ + for (vl = cd->visible; vl != NULL; vl = vl->next) + if (vl->m->slot == no_slot) + generateFunction(pt, vl->m, vl->cd->overs, cd, vl->cd, fp); + + /* The slot functions. */ + for (md = cd->members; md != NULL; md = md->next) + if (cd->iff->type == namespace_iface) + generateOrdinaryFunction(pt,cd,md,fp); + else if (md->slot != no_slot && md->slot != unicode_slot) + generateSlot(pt, cd, NULL, md, fp); + + if (cd->iff->type != namespace_iface && !generating_c) + { + classList *cl; + int need_ptr, need_state; + + /* The cast function. */ + prcode(fp, +"\n" +"\n" +"/* Cast a pointer to a type somewhere in its superclass hierarchy. */\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void *cast_%C(void *, sipWrapperType *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static void *cast_%C(void *ptr,sipWrapperType *targetClass)\n" +"{\n" + ,classFQCName(cd)); + + if (cd->supers != NULL) + prcode(fp, +" void *res;\n" +"\n" + ); + + prcode(fp, +" if (targetClass == sipClass_%C)\n" +" return ptr;\n" + ,classFQCName(cd)); + + for (cl = cd->supers; cl != NULL; cl = cl->next) + { + scopedNameDef *sname = cl->cd->iff->fqcname; + + prcode(fp, +"\n" +" if ((res = sipCast_%C((%S *)(%S *)ptr,targetClass)) != NULL)\n" +" return res;\n" + ,sname,sname,classFQCName(cd)); + } + + prcode(fp, +"\n" +" return NULL;\n" +"}\n" + ); + + /* Generate the release function without compiler warnings. */ + need_ptr = need_state = FALSE; + + if (canCreate(cd) || isPublicDtor(cd)) + { + if (hasShadow(cd)) + need_ptr = need_state = TRUE; + else if (isPublicDtor(cd)) + need_ptr = TRUE; + } + + prcode(fp, +"\n" +"\n" +"/* Call the instance's destructor. */\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void release_%C(void *, int);}\n" + , classFQCName(cd)); + + prcode(fp, +"static void release_%C(void *%s,int%s)\n" +"{\n" + , classFQCName(cd), (need_ptr ? "ptr" : ""), (need_state ? " state" : "")); + + /* + * If there is an explicit public dtor then assume there is + * some way to call it which we haven't worked out (because we + * don't fully understand C++). + */ + if (canCreate(cd) || isPublicDtor(cd)) + { + int rgil = ((release_gil || isReleaseGILDtor(cd)) && !isHoldGILDtor(cd)); + + if (rgil) + prcode(fp, +" Py_BEGIN_ALLOW_THREADS\n" +"\n" + ); + + if (hasShadow(cd)) + { + prcode(fp, +" if (state & SIP_DERIVED_CLASS)\n" +" delete reinterpret_cast<sip%C *>(ptr);\n" + , classFQCName(cd)); + + if (isPublicDtor(cd)) + prcode(fp, +" else\n" +" delete reinterpret_cast<%U *>(ptr);\n" + , cd); + } + else if (isPublicDtor(cd)) + prcode(fp, +" delete reinterpret_cast<%U *>(ptr);\n" + , cd); + + if (rgil) + prcode(fp, +"\n" +" Py_END_ALLOW_THREADS\n" + ); + } + + prcode(fp, +"}\n" + ); + } + + /* The traverse function. */ + if (cd->travcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static int traverse_%C(void *, visitproc, void *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static int traverse_%C(void *sipCppV,visitproc sipVisit,void *sipArg)\n" +"{\n" +" ", classFQCName(cd)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" int sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->travcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + /* The clear function. */ + if (cd->clearcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static int clear_%C(void *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static int clear_%C(void *sipCppV)\n" +"{\n" +" ", classFQCName(cd)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" int sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->clearcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + /* The buffer interface functions. */ + if (cd->readbufcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static SIP_SSIZE_T getreadbuffer_%C(PyObject *, void *, SIP_SSIZE_T, void **);}\n" + , classFQCName(cd)); + + prcode(fp, +"static SIP_SSIZE_T getreadbuffer_%C(PyObject *%s, void *sipCppV, SIP_SSIZE_T %s, void **%s)\n" +"{\n" +" ", classFQCName(cd) + , argName("sipSelf", cd->readbufcode) + , argName("sipSegment", cd->readbufcode) + , argName("sipPtrPtr", cd->readbufcode)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" SIP_SSIZE_T sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->readbufcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + if (cd->writebufcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static SIP_SSIZE_T getwritebuffer_%C(PyObject *, void *, SIP_SSIZE_T, void **);}\n" + , classFQCName(cd)); + + prcode(fp, +"static SIP_SSIZE_T getwritebuffer_%C(PyObject *%s, void *sipCppV, SIP_SSIZE_T %s, void **%s)\n" +"{\n" +" ", classFQCName(cd) + , argName("sipSelf", cd->writebufcode) + , argName("sipSegment", cd->writebufcode) + , argName("sipPtrPtr", cd->writebufcode)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" SIP_SSIZE_T sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->writebufcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + if (cd->segcountcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static SIP_SSIZE_T getsegcount_%C(PyObject *, void *, SIP_SSIZE_T *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static SIP_SSIZE_T getsegcount_%C(PyObject *%s, void *sipCppV, SIP_SSIZE_T *%s)\n" +"{\n" +" ", classFQCName(cd) + , argName("sipSelf", cd->segcountcode) + , argName("sipLenPtr", cd->segcountcode)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" SIP_SSIZE_T sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->segcountcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + if (cd->charbufcode != NULL) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static SIP_SSIZE_T getcharbuffer_%C(PyObject *, void *, SIP_SSIZE_T, void **);}\n" + , classFQCName(cd)); + + prcode(fp, +"static SIP_SSIZE_T getcharbuffer_%C(PyObject *%s, void *sipCppV, SIP_SSIZE_T %s, void **%s)\n" +"{\n" +" ", classFQCName(cd) + , argName("sipSelf", cd->charbufcode) + , argName("sipSegment", cd->charbufcode) + , argName("sipPtrPtr", cd->charbufcode)); + + generateClassFromVoid(cd, "sipCpp", "sipCppV", fp); + + prcode(fp, ";\n" +" SIP_SSIZE_T sipRes;\n" +"\n" + ); + + generateCppCodeBlock(cd->charbufcode, fp); + + prcode(fp, +"\n" +" return sipRes;\n" +"}\n" + ); + } + + /* The dealloc function. */ + if (needDealloc(cd)) + { + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void dealloc_%C(sipWrapper *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static void dealloc_%C(sipWrapper *sipSelf)\n" +"{\n" + ,classFQCName(cd)); + + if (tracing) + prcode(fp, +" sipTrace(SIP_TRACE_DEALLOCS,\"dealloc_%C()\\n\");\n" +"\n" + ,classFQCName(cd)); + + /* Disable the virtual handlers. */ + if (hasShadow(cd)) + prcode(fp, +" if (sipIsDerived(sipSelf))\n" +" reinterpret_cast<sip%C *>(sipSelf->u.cppPtr)->sipPySelf = NULL;\n" +"\n" + ,classFQCName(cd)); + + if (generating_c || isPublicDtor(cd) || (hasShadow(cd) && isProtectedDtor(cd))) + { + prcode(fp, +" if (sipIsPyOwned(sipSelf))\n" +" {\n" + ); + + if (cd->dealloccode != NULL) + { + if (usedInCode(cd->dealloccode, "sipCpp")) + { + prcode(fp, +" "); + + generateClassFromVoid(cd, "sipCpp", "sipSelf->u.cppPtr", fp); + + prcode(fp, ";\n" + ); + } + + generateCppCodeBlock(cd->dealloccode,fp); + + prcode(fp, +"\n" + ); + } + + if (isDelayedDtor(cd)) + prcode(fp, +" sipAddDelayedDtor(sipSelf);\n" + ); + else if (generating_c) + prcode(fp, +" sipFree(sipSelf->u.cppPtr);\n" + ); + else + prcode(fp, +" release_%C(sipSelf->u.cppPtr,%s);\n" + , classFQCName(cd), (hasShadow(cd) ? "sipSelf->flags" : "0")); + + prcode(fp, +" }\n" + ); + } + + prcode(fp, +"}\n" + ); + } + + /* The type initialisation function. */ + if (canCreate(cd)) + generateTypeInit(pt, cd, fp); +} + + +/* + * Generate the shadow (derived) class code. + */ +static void generateShadowCode(sipSpec *pt,classDef *cd,FILE *fp) +{ + int nrVirts, virtNr; + virtOverDef *vod; + ctorDef *ct; + + nrVirts = countVirtuals(cd); + + /* Generate the wrapper class constructors. */ + + for (ct = cd->ctors; ct != NULL; ct = ct->next) + { + char *prefix; + int a; + ctorDef *dct; + + if (isPrivateCtor(ct)) + continue; + + if (ct->cppsig == NULL) + continue; + + /* Check we haven't already handled this C++ signature. */ + for (dct = cd->ctors; dct != ct; dct = dct->next) + if (dct->cppsig != NULL && sameSignature(dct->cppsig, ct->cppsig, TRUE)) + break; + + if (dct != ct) + continue; + + prcode(fp, +"\n" +"sip%C::sip%C(",classFQCName(cd),classFQCName(cd)); + + generateArgs(ct->cppsig,Definition,fp); + + prcode(fp,")%X: %S(",ct->exceptions,classFQCName(cd)); + + prefix = ""; + + for (a = 0; a < ct->cppsig->nrArgs; ++a) + { + prcode(fp,"%sa%d",prefix,a); + prefix = ","; + } + + prcode(fp,"), sipPySelf(0)\n" +"{\n" + ); + + if (tracing) + { + prcode(fp, +" sipTrace(SIP_TRACE_CTORS,\"sip%C::sip%C(",classFQCName(cd),classFQCName(cd)); + generateArgs(ct->cppsig,Declaration,fp); + prcode(fp,")%X (this=0x%%08x)\\n\",this);\n" +"\n" + ,ct->exceptions); + } + + prcode(fp, +" sipCommonCtor(%s,%d);\n" +"}\n" + ,(nrVirts > 0 ? "sipPyMethods" : "NULL"),nrVirts); + } + + /* The destructor. */ + + if (!isPrivateDtor(cd)) + { + prcode(fp, +"\n" +"sip%C::~sip%C()%X\n" +"{\n" + ,classFQCName(cd),classFQCName(cd),cd->dtorexceptions); + + if (tracing) + prcode(fp, +" sipTrace(SIP_TRACE_DTORS,\"sip%C::~sip%C()%X (this=0x%%08x)\\n\",this);\n" +"\n" + ,classFQCName(cd),classFQCName(cd),cd->dtorexceptions); + + if (cd->dtorcode != NULL) + generateCppCodeBlock(cd->dtorcode,fp); + + prcode(fp, +" sipCommonDtor(sipPySelf);\n" +"}\n" + ); + } + + /* The metacall method if required. */ + if (isQObjectSubClass(cd) && optQ_OBJECT4(pt)) + { + prcode(fp, +"\n" +"const QMetaObject *sip%C::metaObject() const\n" +"{\n" +" return sip_%s_qt_metaobject(sipPySelf,sipClass_%C,%S::metaObject());\n" +"}\n" +"\n" +"int sip%C::qt_metacall(QMetaObject::Call _c,int _id,void **_a)\n" +"{\n" +" sip%C::metaObject();\n" +"\n" +" _id = %S::qt_metacall(_c,_id,_a);\n" +"\n" +" if (_id >= 0)\n" +" {\n" +" SIP_BLOCK_THREADS\n" +" _id = sip_%s_qt_metacall(sipPySelf,sipClass_%C,_c,_id,_a);\n" +" SIP_UNBLOCK_THREADS\n" +" }\n" +"\n" +" return _id;\n" +"}\n" + , classFQCName(cd) + , pt->module->name, classFQCName(cd), classFQCName(cd) + , classFQCName(cd) + , classFQCName(cd) + , classFQCName(cd) + , pt->module->name, classFQCName(cd)); + } + + /* Generate the virtual catchers. */ + + virtNr = 0; + + for (vod = cd->vmembers; vod != NULL; vod = vod->next) + { + overDef *od = &vod->o; + virtOverDef *dvod; + + if (isPrivate(od)) + continue; + + /* Check we haven't already handled this C++ signature. */ + for (dvod = cd->vmembers; dvod != vod; dvod = dvod->next) + if (strcmp(dvod->o.cppname,od->cppname) == 0 && sameSignature(dvod->o.cppsig,od->cppsig,TRUE)) + break; + + if (dvod != vod) + continue; + + generateVirtualCatcher(pt,cd,virtNr++,vod,fp); + } + + /* Generate the wrapper around each protected member function. */ + + generateProtectedDefinitions(cd,fp); + + /* Generate the emitters if needed. */ + if (!optNoEmitters(pt)) + generateEmitters(pt, cd, fp); +} + + +/* + * Generate the emitter functions. + */ +static void generateEmitters(sipSpec *pt, classDef *cd, FILE *fp) +{ + int noIntro; + visibleList *vl; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + for (od = vl->cd->overs; od != NULL; od = od->next) + if (od->common == vl->m && isSignal(od)) + { + generateEmitter(pt,cd,vl,fp); + break; + } + } + + /* Generate the table of signals to support fan-outs. */ + + noIntro = TRUE; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + for (od = vl->cd->overs; od != NULL; od = od->next) + if (od->common == vl->m && isSignal(od)) + { + if (noIntro) + { + setHasSigSlots(cd); + + prcode(fp, +"\n" +"static sipQtSignal signals_%C[] = {\n" + ,classFQCName(cd)); + + noIntro = FALSE; + } + + prcode(fp, +" {%N, %C_emit_%s},\n" + ,vl->m->pyname,classFQCName(cd),vl->m->pyname->text); + + break; + } + } + + if (!noIntro) + prcode(fp, +" {NULL, NULL}\n" +"};\n" + ); +} + + +/* + * Generate the protected enums for a class. + */ +static void generateProtectedEnums(sipSpec *pt,classDef *cd,FILE *fp) +{ + enumDef *ed; + + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + char *eol; + enumMemberDef *emd; + + /* Ignore unless this class is the publisher. */ + if (cd != ed->pcd) + continue; + + prcode(fp, +"\n" +" /* Expose this protected enum. */\n" +" enum"); + + if (ed->fqcname != NULL) + prcode(fp," sip%s",scopedNameTail(ed->fqcname)); + + prcode(fp," {"); + + eol = "\n"; + + for (emd = ed->members; emd != NULL; emd = emd->next) + { + prcode(fp,"%s" +" %s = %S::%s",eol,emd->cname,classFQCName(ed->ecd),emd->cname); + + eol = ",\n"; + } + + prcode(fp,"\n" +" };\n" + ); + } +} + + +/* + * Generate the catcher for a virtual function. + */ +static void generateVirtualCatcher(sipSpec *pt, classDef *cd, int virtNr, + virtOverDef *vod, FILE *fp) +{ + overDef *od = &vod->o; + virtHandlerDef *vhd = od->virthandler; + argDef *res, *ad; + int a; + + normaliseArgs(od->cppsig); + + res = &od->cppsig->result; + + if (res->atype == void_type && res->nrderefs == 0) + res = NULL; + + prcode(fp, +"\n"); + + generateBaseType(&od->cppsig->result,fp); + + prcode(fp," sip%C::%O(",classFQCName(cd),od); + generateArgs(od->cppsig,Definition,fp); + prcode(fp,")%s%X\n" +"{\n" + ,(isConst(od) ? " const" : ""),od->exceptions); + + if (tracing) + { + prcode(fp, +" sipTrace(SIP_TRACE_CATCHERS,\""); + + generateBaseType(&od->cppsig->result,fp); + prcode(fp," sip%C::%O(",classFQCName(cd),od); + generateArgs(od->cppsig,Declaration,fp); + prcode(fp,")%s%X (this=0x%%08x)\\n\",this);\n" +"\n" + ,(isConst(od) ? " const" : ""),od->exceptions); + } + + restoreArgs(od->cppsig); + + if (vhd->module == pt->module) + { + prcode(fp, +" extern "); + + generateBaseType(&od->cppsig->result,fp); + + prcode(fp," sipVH_%s_%d(sip_gilstate_t,PyObject *",vhd->module->name,vhd->virthandlernr); + } + else + { + prcode(fp, +" typedef "); + + generateBaseType(&od->cppsig->result,fp); + + prcode(fp," (*sipVH_%s_%d)(sip_gilstate_t,PyObject *",vhd->module->name,vhd->virthandlernr); + } + + if (vhd->cppsig->nrArgs > 0) + { + prcode(fp,","); + generateArgs(vhd->cppsig,Declaration,fp); + } + + prcode(fp,");\n" + ); + + if (isNewThread(od)) + prcode(fp, +"\n" +" SIP_BLOCK_THREADS\n" + ); + + prcode(fp, +"\n" +" sip_gilstate_t sipGILState;\n" +" PyObject *meth;\n" +"\n" +" meth = sipIsPyMethod(&sipGILState,"); + + if (isConst(od)) + prcode(fp,"const_cast<sipMethodCache *>("); + + prcode(fp,"&sipPyMethods[%d]",virtNr); + + if (isConst(od)) + prcode(fp,")"); + + prcode(fp,",sipPySelf,"); + + if (isAbstract(od)) + prcode(fp,"%N",cd->iff->name); + else + prcode(fp,"NULL"); + + prcode(fp,",%N);\n" +"\n" + ,od->common->pyname); + + if (isNewThread(od)) + prcode(fp, +" if (meth)\n" +" {\n" +" sipStartThread();\n" +" "); + else + { + prcode(fp, +" if (!meth)\n" + ); + + if (isAbstract(od)) + generateVirtHandlerErrorReturn(res,fp); + else + { + if (res == NULL) + prcode(fp, +" {\n" +" "); + else + prcode(fp, +" return "); + + generateUnambiguousClass(cd,vod->scope,fp); + + prcode(fp,"::%O(",od); + + for (a = 0; a < od->cppsig->nrArgs; ++a) + prcode(fp,"%sa%d",(a == 0 ? "" : ","),a); + + prcode(fp,");\n" + ); + + if (res == NULL) + prcode(fp, +" return;\n" +" }\n" + ); + } + + prcode(fp, +"\n" +" "); + + if (res != NULL) + prcode(fp,"return "); + } + + if (vhd->module == pt->module) + prcode(fp,"sipVH_%s_%d",vhd->module->name,vhd->virthandlernr); + else + prcode(fp,"((sipVH_%s_%d)(sipModuleAPI_%s_%s->em_virthandlers[%d]))",vhd->module->name,vhd->virthandlernr,pt->module->name,vhd->module->name,vhd->virthandlernr); + + prcode(fp,"(sipGILState,meth"); + + ad = od->cppsig->args; + + for (a = 0; a < od->cppsig->nrArgs; ++a) + { + if (ad->atype == class_type && isProtectedClass(ad->u.cd)) + prcode(fp,",static_cast<%U *>(a%d)",ad->u.cd,a); + else if (ad->atype == enum_type && isProtectedEnum(ad->u.ed)) + prcode(fp, ",(%E)a%d", ad->u.ed, a); + else + prcode(fp,",a%d",a); + + ++ad; + } + + prcode(fp,");\n" + ); + + if (isNewThread(od)) + prcode(fp, +" sipEndThread();\n" +" }\n" +"\n" +" SIP_UNBLOCK_THREADS\n" + ); + + prcode(fp, +"}\n" + ); +} + + +/* + * Generate the scope of the near class of a virtual taking duplicate + * super-classes into account. + */ +static void generateUnambiguousClass(classDef *cd,classDef *scope,FILE *fp) +{ + mroDef *mro; + + /* See if the near class has a duplicate. */ + for (mro = cd->mro; mro != NULL; mro = mro->next) + if (mro->cd == scope) + { + if (hasDuplicateSuper(mro)) + { + mroDef *guardc; + + /* + * Backtrack to find the class that directly + * sub-classes the duplicated one. This will + * be the one that disambiguates the duplicated + * one. + */ + guardc = mro; + + while (guardc != cd->mro) + { + mroDef *sub; + classList *cl; + + for (sub = cd->mro; sub->next != guardc; sub = sub->next) + ; + + for (cl = sub->cd->supers; cl != NULL; cl = cl->next) + if (cl->cd == mro->cd) + { + prcode(fp,"%S",classFQCName(sub->cd)); + + return; + } + + /* Try the previous one. */ + guardc = sub; + } + } + + break; + } + + /* If we got here there is nothing to worry about. */ + prcode(fp,"%S",classFQCName(scope)); +} + + +/* + * Generate a cast to zero. + */ +static void generateCastZero(argDef *ad,FILE *fp) +{ + if (ad->atype == enum_type) + prcode(fp,"(%E)",ad->u.ed); + + prcode(fp,"0"); +} + + +/* + * Generate the return statement for a virtual handler when there has been an + * error (ie. there is nothing sensible to return). + */ +static void generateVirtHandlerErrorReturn(argDef *res,FILE *fp) +{ + prcode(fp, +" return"); + + if (res == NULL) + { + prcode(fp,";\n" + ); + + return; + } + + prcode(fp," "); + + if (res->atype == mapped_type && res->nrderefs == 0) + { + argDef res_noconstref; + + /* + * We don't know anything about the mapped type so we just hope + * is has a default ctor. + */ + + if (isReference(res)) + prcode(fp,"*new "); + + res_noconstref = *res; + resetIsConstArg(&res_noconstref); + resetIsReference(&res_noconstref); + prcode(fp,"%B()",&res_noconstref); + } + else if (res->atype == class_type && res->nrderefs == 0) + { + ctorDef *ct = res->u.cd->defctor; + + /* + * If we don't have a suitable ctor then the generated code + * will issue an error message. + */ + if (ct != NULL && isPublicCtor(ct) && ct->cppsig != NULL) + { + argDef res_noconstref; + + /* + * If this is a badly designed class. We can only + * generate correct code by leaking memory. + */ + if (isReference(res)) + prcode(fp,"*new "); + + res_noconstref = *res; + resetIsConstArg(&res_noconstref); + resetIsReference(&res_noconstref); + prcode(fp,"%B",&res_noconstref); + + generateCallDefaultCtor(ct,fp); + } + else + { + fatalScopedName(classFQCName(res->u.cd)); + fatal(" must have a default constructor\n"); + } + } + else + generateCastZero(res,fp); + + prcode(fp,";\n" + ); +} + + +/* + * Generate the call to a default ctor. + */ +static void generateCallDefaultCtor(ctorDef *ct, FILE *fp) +{ + int a; + + prcode(fp, "("); + + for (a = 0; a < ct->cppsig->nrArgs; ++a) + { + argDef *ad = &ct->cppsig->args[a]; + + if (ad->defval != NULL) + break; + + if (a > 0) + prcode(fp, ","); + + /* + * Do what we can to provide type information to the compiler. + */ + if (ad->atype == class_type && ad->nrderefs > 0 && !isReference(ad)) + prcode(fp, "static_cast<%B>(0)", ad); + else if (ad->atype == enum_type) + prcode(fp, "static_cast<%E>(0)", ad->u.ed); + else if (ad->atype == float_type || ad->atype == cfloat_type) + prcode(fp, "0.0F"); + else if (ad->atype == double_type || ad->atype == cdouble_type) + prcode(fp, "0.0"); + else if (ad->atype == uint_type) + prcode(fp, "0U"); + else if (ad->atype == long_type || ad->atype == longlong_type) + prcode(fp, "0L"); + else if (ad->atype == ulong_type || ad->atype == ulonglong_type) + prcode(fp, "0UL"); + else if ((ad->atype == ustring_type || ad->atype == sstring_type || ad->atype == string_type) && ad->nrderefs == 0) + prcode(fp, "'\\0'"); + else if (ad->atype == wstring_type && ad->nrderefs == 0) + prcode(fp, "L'\\0'"); + else + prcode(fp, "0"); + } + + prcode(fp, ")"); +} + + +/* + * Generate the emitter function for a signal. + */ +static void generateEmitter(sipSpec *pt,classDef *cd,visibleList *vl,FILE *fp) +{ + char *pname = vl->m->pyname->text; + overDef *od; + + prcode(fp, +"\n" +"int sip%C::sipEmit_%s(PyObject *sipArgs)\n" +"{\n" +" int sipArgsParsed = 0;\n" + ,classFQCName(cd),pname); + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + int rgil = ((release_gil || isReleaseGIL(od)) && !isHoldGIL(od)); + + if (od->common != vl->m || !isSignal(od)) + continue; + + /* + * Generate the code that parses the args and emits the + * appropriate overloaded signal. + */ + prcode(fp, +"\n" +" {\n" + ); + + generateArgParser(pt, &od->pysig, cd, NULL, NULL, FALSE, fp); + + prcode(fp, +" {\n" + ); + + if (rgil) + prcode(fp, +" Py_BEGIN_ALLOW_THREADS\n" + ); + + prcode(fp, +" emit %s(" + ,od->cppname); + + generateArgs(od->cppsig,Call,fp); + + prcode(fp,");\n" + ); + + if (rgil) + prcode(fp, +" Py_END_ALLOW_THREADS\n" + ); + + deleteTemps(&od->pysig, fp); + + prcode(fp, +"\n" +" return 0;\n" +" }\n" +" }\n" + ); + } + + prcode(fp, +"\n" +" sipNoMethod(sipArgsParsed,%N,%N);\n" +"\n" +" return -1;\n" +"}\n" +"\n" + , cd->iff->name, vl->m->pyname); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static int %C_emit_%s(sipWrapper *, PyObject *);}\n" + , classFQCName(cd), pname); + + prcode(fp, +"static int %C_emit_%s(sipWrapper *w,PyObject *sipArgs)\n" +"{\n" +" sip%C *ptr = reinterpret_cast<sip%C *>(sipGetComplexCppPtr(w));\n" +"\n" +" return (ptr ? ptr->sipEmit_%s(sipArgs) : -1);\n" +"}\n" + ,classFQCName(cd),pname + ,classFQCName(cd),classFQCName(cd) + ,pname); +} + + +/* + * Generate the declarations of the protected wrapper functions for a class. + */ + +static void generateProtectedDeclarations(classDef *cd,FILE *fp) +{ + int noIntro; + visibleList *vl; + + noIntro = TRUE; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + if (vl->m->slot != no_slot) + continue; + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + if (od->common != vl->m || !isProtected(od)) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +" /*\n" +" * There is a public method for every protected method visible from\n" +" * this class.\n" +" */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" "); + + if (isStatic(od)) + prcode(fp,"static "); + + generateBaseType(&od->cppsig->result,fp); + + if (!isStatic(od) && !isAbstract(od) && (isVirtual(od) || isVirtualReimp(od))) + { + prcode(fp, " sipProtectVirt_%s(bool", od->cppname); + + if (od->cppsig->nrArgs > 0) + prcode(fp, ","); + } + else + prcode(fp, " sipProtect_%s(", od->cppname); + + generateArgs(od->cppsig,Declaration,fp); + prcode(fp,")%s;\n" + ,(isConst(od) ? " const" : "")); + } + } +} + + +/* + * Generate the definitions of the protected wrapper functions for a class. + */ +static void generateProtectedDefinitions(classDef *cd,FILE *fp) +{ + visibleList *vl; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + if (vl->m->slot != no_slot) + continue; + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + char *mname = od->cppname; + int parens; + argDef *res; + + if (od->common != vl->m || !isProtected(od)) + continue; + + prcode(fp, +"\n" + ); + + generateBaseType(&od->cppsig->result,fp); + + if (!isStatic(od) && !isAbstract(od) && (isVirtual(od) || isVirtualReimp(od))) + { + prcode(fp, " sip%C::sipProtectVirt_%s(bool sipSelfWasArg", classFQCName(cd), mname); + + if (od->cppsig->nrArgs > 0) + prcode(fp, ","); + } + else + prcode(fp, " sip%C::sipProtect_%s(", classFQCName(cd), mname); + + generateArgs(od->cppsig,Definition,fp); + prcode(fp,")%s\n" +"{\n" + ,(isConst(od) ? " const" : "")); + + parens = 1; + + res = &od->cppsig->result; + + if (res->atype == void_type && res->nrderefs == 0) + prcode(fp, +" "); + else + { + prcode(fp, +" return "); + + if (res->atype == class_type && isProtectedClass(res->u.cd)) + { + prcode(fp,"static_cast<%U *>(",res->u.cd); + ++parens; + } + else if (res->atype == enum_type && isProtectedEnum(res->u.ed)) + /* + * One or two older compilers can't + * handle a static_cast here so we + * revert to a C-style cast. + */ + prcode(fp,"(%E)",res->u.ed); + } + + if (!isAbstract(od)) + if (isVirtual(od) || isVirtualReimp(od)) + { + prcode(fp, "(sipSelfWasArg ? %S::%s(", classFQCName(vl->cd), mname); + + generateProtectedCallArgs(od, fp); + + prcode(fp, ") : "); + ++parens; + } + else + prcode(fp, "%S::", classFQCName(vl->cd)); + + prcode(fp,"%s(",mname); + + generateProtectedCallArgs(od, fp); + + while (parens--) + prcode(fp,")"); + + prcode(fp,";\n" +"}\n" + ); + } + } +} + + +/* + * Generate the arguments for a call to a protected method. + */ +static void generateProtectedCallArgs(overDef *od, FILE *fp) +{ + int a; + + for (a = 0; a < od->cppsig->nrArgs; ++a) + { + argDef *ad = &od->cppsig->args[a]; + + if (a > 0) + prcode(fp, ","); + + if (ad->atype == enum_type && isProtectedEnum(ad->u.ed)) + prcode(fp, "(%S)", ad->u.ed->fqcname); + + prcode(fp, "a%d", a); + } +} + + +/* + * Generate the function that does most of the work to handle a particular + * virtual function. + */ +static void generateVirtualHandler(sipSpec *pt,virtHandlerDef *vhd,FILE *fp) +{ + int a, nrvals, copy, isref; + argDef *res, res_noconstref; + + res = &vhd->cppsig->result; + + copy = isref = FALSE; + + if (res->atype == void_type && res->nrderefs == 0) + res = NULL; + else + { + /* + * If we are returning a reference to an instance then we take care to + * handle Python errors but still return a valid C++ instance. If we + * are returning an instance then we take care to make a local copy of + * the instance returned from Python before the Python object is + * garbage collected and the C++ instance (possibly) destroyed. + */ + if ((res->atype == class_type || res->atype == mapped_type) && res->nrderefs == 0) + if (isReference(res)) + isref = TRUE; + else + copy = TRUE; + + res_noconstref = *res; + resetIsConstArg(&res_noconstref); + resetIsReference(&res_noconstref); + } + + prcode(fp, +"\n" + ); + + generateBaseType(&vhd->cppsig->result, fp); + + prcode(fp," sipVH_%s_%d(sip_gilstate_t sipGILState,PyObject *sipMethod" + ,pt->module->name,vhd->virthandlernr); + + if (vhd->cppsig->nrArgs > 0) + { + prcode(fp,","); + + generateArgs(vhd->cppsig, Definition, fp); + } + + prcode(fp,")\n" +"{\n" + ); + + if (res != NULL) + { + prcode(fp, " "); + + /* + * wchar_t * return values are always on the heap. To reduce memory + * leaks we keep the last result around until we have a new one. This + * means that ownership of the return value stays with the function + * returning it - which is consistent with how other types work, even + * thought it may not be what's required in all cases. + */ + if (res->atype == wstring_type && res->nrderefs == 1) + prcode(fp, "static "); + + generateBaseType(&res_noconstref,fp); + + prcode(fp," %ssipRes",(isref ? "*" : "")); + + if (copy && res->atype == class_type && res->nrderefs == 0) + { + ctorDef *ct = res->u.cd->defctor; + + if (ct != NULL && isPublicCtor(ct) && ct->cppsig != NULL && ct->cppsig->nrArgs > 0 && ct->cppsig->args[0].defval == NULL) + generateCallDefaultCtor(ct,fp); + } + else if (!copy) + { + /* + * We initialise the result to try and suppress a + * compiler warning. + */ + prcode(fp," = "); + generateCastZero(res,fp); + } + + prcode(fp,";\n" + ); + + if (res->atype == wstring_type && res->nrderefs == 1) + prcode(fp, +"\n" +" if (sipRes)\n" +" {\n" +" // Return any previous result to the heap.\n" +" sipFree(%s);\n" +" sipRes = 0;\n" +" }\n" +"\n" + , (isConstArg(res) ? "const_cast<wchar_t *>(sipRes)" : "sipRes")); + } + + if (vhd->virtcode != NULL) + { + int error_flag = needErrorFlag(vhd->virtcode); + + if (error_flag) + prcode(fp, +" int sipIsErr = 0;\n" + ); + + prcode(fp, +"\n" + ); + + generateCppCodeBlock(vhd->virtcode,fp); + + if (error_flag) + prcode(fp, +"\n" +" if (sipIsErr)\n" +" PyErr_Print();\n" + ); + + prcode(fp, +"\n" +" Py_DECREF(sipMethod);\n" +"\n" +" SIP_RELEASE_GIL(sipGILState)\n" + ); + + if (res != NULL) + prcode(fp, +"\n" +" return sipRes;\n" + ); + + prcode(fp, +"}\n" + ); + + return; + } + + /* See how many values we expect. */ + nrvals = (res != NULL ? 1 : 0); + + for (a = 0; a < vhd->pysig->nrArgs; ++a) + if (isOutArg(&vhd->pysig->args[a])) + ++nrvals; + + if (copy) + { + prcode(fp, +" "); + + generateBaseType(&res_noconstref,fp); + + prcode(fp," *sipResOrig;\n"); + + if (res->atype == class_type && res->u.cd->convtocode != NULL) + prcode(fp, +" int sipResState;\n" + ); + } + + /* Call the method. */ + prcode(fp, +" PyObject *sipResObj = sipCallMethod(0,sipMethod,"); + + generateTupleBuilder(vhd->pysig, fp); + + prcode(fp,");\n" +"\n" +" %s (!sipResObj || sipParseResult(0,sipMethod,sipResObj,\"",(isref ? "int sipIsErr =" : "if")); + + /* Build the format string. */ + if (nrvals == 0) + prcode(fp,"Z"); + else + { + if (nrvals > 1) + prcode(fp,"("); + + if (res != NULL) + prcode(fp, "%s", getParseResultFormat(res, TRUE, isTransferVH(vhd))); + + for (a = 0; a < vhd->pysig->nrArgs; ++a) + { + argDef *ad = &vhd->pysig->args[a]; + + if (isOutArg(ad)) + prcode(fp, "%s", getParseResultFormat(ad, FALSE, FALSE)); + } + + if (nrvals > 1) + prcode(fp,")"); + } + + prcode(fp,"\""); + + /* Pass the destination pointers. */ + if (res != NULL) + { + generateParseResultExtraArgs(res, TRUE, fp); + prcode(fp,",&sipRes%s",(copy ? "Orig" : "")); + } + + for (a = 0; a < vhd->pysig->nrArgs; ++a) + { + argDef *ad = &vhd->pysig->args[a]; + + if (isOutArg(ad)) + { + generateParseResultExtraArgs(ad, FALSE, fp); + prcode(fp,",%sa%d",(isReference(ad) ? "&" : ""),a); + } + } + + if (isref) + prcode(fp,") < 0);\n" +"\n" +" if (sipIsErr)\n" + ); + else + prcode(fp,") < 0)\n" + ); + + prcode(fp, +" PyErr_Print();\n" + ); + + /* Make a copy if needed. */ + if (copy) + { + prcode(fp, +" else\n" +" {\n" +" sipRes = *sipResOrig;\n" + ); + + if (res->atype == mapped_type) + prcode(fp, +" delete sipResOrig;\n" + ); + else if (res->atype == class_type && res->u.cd->convtocode != NULL) + prcode(fp, +" sipReleaseInstance(sipResOrig,sipClass_%C,sipResState);\n" + , classFQCName(res->u.cd)); + + prcode(fp, +" }\n" + ); + } + + prcode(fp, +"\n" +" Py_XDECREF(sipResObj);\n" +" Py_DECREF(sipMethod);\n" +"\n" +" SIP_RELEASE_GIL(sipGILState)\n" + ); + + if (res != NULL) + { + if (isref) + { + prcode(fp, +"\n" +" if (sipIsErr)\n" + ); + + generateVirtHandlerErrorReturn(res,fp); + } + + prcode(fp, +"\n" +" return %ssipRes;\n" + ,(isref ? "*" : "")); + } + + prcode(fp, +"}\n" + ); +} + + +/* + * Generate the extra arguments needed by sipParseResult() for a particular + * type. + */ +static void generateParseResultExtraArgs(argDef *ad, int isres, FILE *fp) +{ + switch (ad->atype) + { + case mapped_type: + prcode(fp, ",sipMappedType_%T", ad); + break; + + case class_type: + prcode(fp, ",sipClass_%C", classFQCName(ad->u.cd)); + + if (isres && ad->nrderefs == 0 && ad->u.cd->convtocode != NULL && !isReference(ad)) + prcode(fp, ",&sipResState"); + + break; + + case pytuple_type: + prcode(fp,",&PyTuple_Type"); + break; + + case pylist_type: + prcode(fp,",&PyList_Type"); + break; + + case pydict_type: + prcode(fp,",&PyDict_Type"); + break; + + case pyslice_type: + prcode(fp,",&PySlice_Type"); + break; + + case pytype_type: + prcode(fp,",&PyType_Type"); + break; + + case enum_type: + if (ad->u.ed->fqcname != NULL) + prcode(fp,",sipEnum_%C",ad->u.ed->fqcname); + break; + } +} + + +/* + * Return the format characters used by sipParseResult() for a particular type. + */ +static const char *getParseResultFormat(argDef *ad, int isres, int xfervh) +{ + switch (ad->atype) + { + case mapped_type: + { + static const char *s[] = { + "D0", "D1", "D2", "D3", + "D4", "D5", "D6", "D7" + }; + + int f = 0x04; + + if (isres && ad->nrderefs == 0) + f |= 0x01; + + if (isres && xfervh) + f |= 0x02; + + return s[f]; + } + + case class_type: + { + static char s[] = "C?"; + + int f = 0x04; + + if (isres && ad->nrderefs == 0) + { + f |= 0x01; + + if (ad->u.cd->convtocode != NULL) + { + f &= ~0x04; + + /* + * If it is a reference then we are + * going to return the dereference. To + * make sure it remains valid we can + * either leak the temporary from the + * %ConvertToCode or we can suppress + * the %ConvertToCode. We choose the + * latter. + */ + if (isReference(ad)) + f |= 0x10; + } + } + + if (isres && xfervh) + f |= 0x02; + + s[1] = '0' + f; + + return s; + } + + case bool_type: + case cbool_type: + return "b"; + + case sstring_type: + case ustring_type: + case string_type: + return ((ad->nrderefs == 0) ? "c" : "s"); + + case wstring_type: + return ((ad->nrderefs == 0) ? "w" : "x"); + + case enum_type: + return ((ad->u.ed->fqcname != NULL) ? "E" : "e"); + + case ushort_type: + return "t"; + + case short_type: + return "h"; + + case int_type: + case cint_type: + return "i"; + + case uint_type: + return "u"; + + case long_type: + return "l"; + + case ulong_type: + return "m"; + + case longlong_type: + return "n"; + + case ulonglong_type: + return "o"; + + case void_type: + case struct_type: + return "V"; + + case float_type: + case cfloat_type: + return "f"; + + case double_type: + case cdouble_type: + return "d"; + + case pyobject_type: + return "O"; + + case pytuple_type: + case pylist_type: + case pydict_type: + case pyslice_type: + case pytype_type: + return (isAllowNone(ad) ? "N" : "T"); + } + + /* We should never get here. */ + return " "; +} + + +/* + * Generate the code to build a tuple of Python arguments. + */ +static void generateTupleBuilder(signatureDef *sd,FILE *fp) +{ + int a, arraylenarg; + + prcode(fp,"\""); + + for (a = 0; a < sd->nrArgs; ++a) + { + char *fmt = ""; + argDef *ad = &sd->args[a]; + + if (!isInArg(ad)) + continue; + + switch (ad->atype) + { + case sstring_type: + case ustring_type: + case string_type: + if (ad->nrderefs == 0 || (ad->nrderefs == 1 && isOutArg(ad))) + fmt = "c"; + else if (isArray(ad)) + fmt = "a"; + else + fmt = "s"; + + break; + + case wstring_type: + if (ad->nrderefs == 0 || (ad->nrderefs == 1 && isOutArg(ad))) + fmt = "w"; + else if (isArray(ad)) + fmt = "A"; + else + fmt = "x"; + + break; + + case bool_type: + case cbool_type: + fmt = "b"; + break; + + case enum_type: + fmt = (ad->u.ed->fqcname != NULL) ? "E" : "e"; + break; + + case cint_type: + fmt = "i"; + break; + + case uint_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "u"; + + break; + + case int_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "i"; + + break; + + case ushort_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "t"; + + break; + + case short_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "h"; + + break; + + case long_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "l"; + + break; + + case ulong_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "m"; + + break; + + case longlong_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "n"; + + break; + + case ulonglong_type: + if (isArraySize(ad)) + arraylenarg = a; + else + fmt = "o"; + + break; + + case struct_type: + case void_type: + fmt = "V"; + break; + + case float_type: + case cfloat_type: + fmt = "f"; + break; + + case double_type: + case cdouble_type: + fmt = "d"; + break; + + case signal_type: + case slot_type: + case slotcon_type: + case slotdis_type: + fmt = "s"; + break; + + case mapped_type: + fmt = "D"; + break; + + case class_type: + fmt = "C"; + break; + + case rxcon_type: + case rxdis_type: + case qobject_type: + fmt = "O"; + break; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + fmt = "S"; + break; + } + + prcode(fp,fmt); + } + + prcode(fp,"\""); + + for (a = 0; a < sd->nrArgs; ++a) + { + int derefs; + argDef *ad = &sd->args[a]; + + if (!isInArg(ad)) + continue; + + derefs = ad->nrderefs; + + switch (ad->atype) + { + case sstring_type: + case ustring_type: + case string_type: + case wstring_type: + if (!(ad->nrderefs == 0 || (ad->nrderefs == 1 && isOutArg(ad)))) + --derefs; + + break; + + case mapped_type: + case class_type: + if (ad->nrderefs > 0) + --derefs; + + break; + + case struct_type: + case void_type: + --derefs; + break; + } + + if (ad->atype == mapped_type || ad->atype == class_type || + ad->atype == rxcon_type || ad->atype == rxdis_type || + ad->atype == qobject_type) + { + prcode(fp,","); + + if (isConstArg(ad)) + prcode(fp,"const_cast<%b *>(",ad); + + if (ad->nrderefs == 0) + prcode(fp,"&"); + else + while (derefs-- != 0) + prcode(fp,"*"); + + prcode(fp,"a%d",a); + + if (isConstArg(ad)) + prcode(fp,")"); + + if (ad->atype == mapped_type) + prcode(fp, ",sipMappedType_%T,NULL", ad); + else if (ad->atype == class_type) + prcode(fp, ",sipClass_%C,NULL", classFQCName(ad->u.cd)); + else + prcode(fp,",sipClass_QObject"); + } + else + { + if (!isArraySize(ad)) + { + prcode(fp,","); + + while (derefs-- != 0) + prcode(fp,"*"); + + prcode(fp,"a%d",a); + } + + if (isArray(ad)) + { + argType astype = sd->args[arraylenarg].atype; + + prcode(fp,",%sa%d",(astype == int_type ? "" : "(int)"),arraylenarg); + } + else if (ad->atype == enum_type && ad->u.ed->fqcname != NULL) + prcode(fp,",sipEnum_%C",ad->u.ed->fqcname); + } + } +} + + +/* + * Generate the class interface #include directives required by either a class + * or a module. + */ +static void generateUsedIncludes(ifaceFileList *iffl, int header, FILE *fp) +{ + int newl = TRUE; + + while (iffl != NULL) + { + if (header == iffl->header) + { + if (newl) + { + prcode(fp, +"\n" + ); + + newl = FALSE; + } + + prcode(fp, +"#include \"sip%s%F.h\"\n" + , iffl->iff->module->name, iffl->iff->fqcname); + } + + iffl = iffl->next; + } + + if (!newl) + prcode(fp, +"\n" + ); +} + + +/* + * Generate the header file for the C++ interface. + */ +static void generateIfaceHeader(sipSpec *pt,ifaceFileDef *iff,char *codeDir) +{ + char *wfile; + char *cmname = iff->module->name; + classDef *cd; + mappedTypeDef *mtd; + exceptionDef *xd; + int genused; + FILE *fp; + + /* Create the header file. */ + + wfile = createIfaceFileName(codeDir,iff,".h"); + fp = createFile(pt,wfile,"Interface header file."); + + prcode(fp, +"\n" +"#ifndef _%s%F_h\n" +"#define _%s%F_h\n" +"\n" + ,cmname,iff->fqcname,cmname,iff->fqcname); + + genused = TRUE; + + for (cd = pt->classes; cd != NULL; cd = cd->next) + if (cd->iff == iff) + { + if (iff->module == pt->module) + generateClassHeader(cd,genused,pt,fp); + else if (!isExternal(cd)) + generateImportedClassHeader(cd,pt,fp); + + genused = FALSE; + } + + genused = TRUE; + + for (mtd = pt->mappedtypes; mtd != NULL; mtd = mtd->next) + if (mtd->iff == iff) + { + if (iff->module == pt->module) + generateMappedTypeHeader(mtd,genused,fp); + else + generateImportedMappedTypeHeader(mtd,pt,fp); + + genused = FALSE; + } + + for (xd = pt->exceptions; xd != NULL; xd = xd->next) + if (xd->iff == iff) + { + generateCppCodeBlock(xd->hdrcode,fp); + + if (xd->exceptionnr >= 0) + { + prcode(fp, +"\n" +"#define sipException_%C sipModuleAPI_%s" + ,iff->fqcname,pt->module->name); + + if (iff->module == pt->module) + prcode(fp,"."); + else + prcode(fp,"_%s->",iff->module->name); + + prcode(fp,"em_exceptions[%d]\n" + ,xd->exceptionnr); + } + } + + prcode(fp, +"\n" +"#endif\n" + ); + + closeFile(fp); + free(wfile); +} + + +/* + * Generate the C++ header code for an imported mapped type. + */ +static void generateImportedMappedTypeHeader(mappedTypeDef *mtd,sipSpec *pt, + FILE *fp) +{ + char *mname = pt->module->name; + char *imname = mtd->iff->module->name; + argDef type; + + generateCppCodeBlock(mtd->hdrcode,fp); + + type.atype = mapped_type; + type.u.mtd = mtd; + type.argflags = 0; + type.name = NULL; + type.nrderefs = 0; + type.defval = NULL; + + prcode(fp, +"\n" +"#define sipMappedType_%T sipModuleAPI_%s_%s->em_mappedtypes[%d]\n" +"#define sipForceConvertTo_%T sipModuleAPI_%s_%s->em_mappedtypes[%d]->mt_fcto\n" +"#define sipConvertFrom_%T sipModuleAPI_%s_%s->em_mappedtypes[%d]->mt_cfrom\n" + ,&type,mname,imname,mtd->mappednr + ,&type,mname,imname,mtd->mappednr + ,&type,mname,imname,mtd->mappednr); +} + + +/* + * Generate the C++ header code for a generated mapped type. + */ +static void generateMappedTypeHeader(mappedTypeDef *mtd,int genused,FILE *fp) +{ + prcode(fp, +"\n" +"\n" + ); + + generateCppCodeBlock(mtd->hdrcode,fp); + + if (genused) + generateUsedIncludes(mtd->iff->used, TRUE, fp); + + prcode(fp, +"\n" +"#define sipMappedType_%T &sipMappedTypeDef_%T\n" +"#define sipForceConvertTo_%T sipMappedTypeDef_%T.mt_fcto\n" +"#define sipConvertFrom_%T sipMappedTypeDef_%T.mt_cfrom\n" +"\n" +"extern sipMappedType sipMappedTypeDef_%T;\n" + ,&mtd->type,&mtd->type + ,&mtd->type,&mtd->type + ,&mtd->type,&mtd->type + ,&mtd->type); +} + + +/* + * Generate the C++ header code for an imported class. + */ +static void generateImportedClassHeader(classDef *cd,sipSpec *pt,FILE *fp) +{ + char *mname = pt->module->name; + char *imname = cd->iff->module->name; + classDef *hcd; + + for (hcd = cd; hcd != NULL; hcd = hcd->ecd) + generateCppCodeBlock(hcd->hdrcode,fp); + + prcode(fp, +"\n" +"#define sipClass_%C sipModuleAPI_%s_%s->em_types[%d]\n" +"#define sipCast_%C sipModuleAPI_%s_%s->em_types[%d]->type->td_cast\n" +"#define sipForceConvertTo_%C sipModuleAPI_%s_%s->em_types[%d]->type->td_fcto\n" + ,classFQCName(cd),mname,imname,cd->classnr + ,classFQCName(cd),mname,imname,cd->classnr + ,classFQCName(cd),mname,imname,cd->classnr); + + generateEnumMacros(pt, cd, fp); +} + + +/* + * Generate the C++ header code for a generated class. + */ +static void generateClassHeader(classDef *cd,int genused,sipSpec *pt,FILE *fp) +{ + char *mname = pt->module->name; + classDef *hcd; + + for (hcd = cd; hcd != NULL; hcd = hcd->ecd) + generateCppCodeBlock(hcd->hdrcode,fp); + + if (genused) + generateUsedIncludes(cd->iff->used, TRUE, fp); + + if (cd->iff->type != namespace_iface) + { + prcode(fp, +"\n" +"#define sipClass_%C sipModuleAPI_%s.em_types[%d]\n" + ,classFQCName(cd),mname,cd->classnr); + + if (!isExternal(cd)) + prcode(fp, +"#define sipCast_%C sipType_%s_%C.td_cast\n" +"#define sipForceConvertTo_%C sipType_%s_%C.td_fcto\n" + , classFQCName(cd), mname, classFQCName(cd) + , classFQCName(cd), mname, classFQCName(cd)); + } + + generateEnumMacros(pt, cd, fp); + + if (!isExternal(cd)) + prcode(fp, +"\n" +"extern sipTypeDef sipType_%s_%C;\n" + , mname, classFQCName(cd)); + + if (hasShadow(cd)) + generateShadowClassDeclaration(pt,cd,fp); +} + + +/* + * Generate the sipEnum_* macros. + */ +static void generateEnumMacros(sipSpec *pt, classDef *cd, FILE *fp) +{ + enumDef *ed; + int noIntro = TRUE; + + for (ed = pt->enums; ed != NULL; ed = ed->next) + { + if (ed->fqcname == NULL || ed->ecd != cd) + continue; + + if (noIntro) + { + prcode(fp, +"\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +"#define sipEnum_%C sipModuleAPI_%s", ed->fqcname, pt->module->name); + + if (pt->module == ed->module) + prcode(fp, "."); + else + prcode(fp, "_%s->", ed->module->name); + + prcode(fp, "em_enums[%d]\n" + , ed->enumnr); + } +} + + +/* + * Generate the shadow class declaration. + */ +static void generateShadowClassDeclaration(sipSpec *pt,classDef *cd,FILE *fp) +{ + int noIntro, nrVirts; + ctorDef *ct; + virtOverDef *vod; + classDef *pcd; + + prcode(fp, +"\n" +"\n" +"class sip%C : public %S\n" +"{\n" +"public:\n" + ,classFQCName(cd),classFQCName(cd)); + + /* Define a shadow class for any protected classes we have. */ + + for (pcd = pt->classes; pcd != NULL; pcd = pcd->next) + { + if (pcd->ecd != cd || !isProtectedClass(pcd)) + continue; + + prcode(fp, +" class sip%s : public %s {};\n" + ,classBaseName(pcd),classBaseName(pcd)); + } + + /* The constructor declarations. */ + + for (ct = cd->ctors; ct != NULL; ct = ct->next) + { + ctorDef *dct; + + if (isPrivateCtor(ct)) + continue; + + if (ct->cppsig == NULL) + continue; + + /* Check we haven't already handled this C++ signature. */ + for (dct = cd->ctors; dct != ct; dct = dct->next) + if (dct->cppsig != NULL && sameSignature(dct->cppsig, ct->cppsig, TRUE)) + break; + + if (dct != ct) + continue; + + prcode(fp, +" sip%C(",classFQCName(cd)); + + generateArgs(ct->cppsig,Declaration,fp); + + prcode(fp,")%X;\n" + ,ct->exceptions); + } + + /* The destructor. */ + + if (!isPrivateDtor(cd)) + prcode(fp, +" %s~sip%C()%X;\n" + ,(cd->vmembers != NULL ? "virtual " : ""),classFQCName(cd),cd->dtorexceptions); + + /* The metacall methods if required. */ + if (isQObjectSubClass(cd) && optQ_OBJECT4(pt)) + prcode(fp, +"\n" +" const QMetaObject *metaObject() const;\n" +" int qt_metacall(QMetaObject::Call,int,void **);\n" + ); + + /* The exposure of protected enums. */ + + generateProtectedEnums(pt,cd,fp); + + /* The wrapper around each protected member function. */ + + generateProtectedDeclarations(cd,fp); + + /* The public wrapper around each signal emitter. */ + if (!optNoEmitters(pt)) + { + visibleList *vl; + + noIntro = TRUE; + + for (vl = cd->visible; vl != NULL; vl = vl->next) + { + overDef *od; + + if (vl->m->slot != no_slot) + continue; + + for (od = vl->cd->overs; od != NULL; od = od->next) + { + if (od->common != vl->m || !isSignal(od)) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +" /*\n" +" * There is a public method for every Qt signal that can be emitted\n" +" * by this object. This function is called by Python to emit the\n" +" * signal.\n" +" */\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" int sipEmit_%s(PyObject *);\n" + ,vl->m->pyname->text); + + break; + } + } + } + + /* The catcher around each virtual function in the hierarchy. */ + noIntro = TRUE; + + for (vod = cd->vmembers; vod != NULL; vod = vod->next) + { + overDef *od = &vod->o; + virtOverDef *dvod; + + if (isPrivate(od)) + continue; + + /* Check we haven't already handled this C++ signature. */ + for (dvod = cd->vmembers; dvod != vod; dvod = dvod->next) + if (strcmp(dvod->o.cppname,od->cppname) == 0 && sameSignature(dvod->o.cppsig,od->cppsig,TRUE)) + break; + + if (dvod != vod) + continue; + + if (noIntro) + { + prcode(fp, +"\n" +" /*\n" +" * There is a protected method for every virtual method visible from\n" +" * this class.\n" +" */\n" +"protected:\n" + ); + + noIntro = FALSE; + } + + prcode(fp, +" "); + + prOverloadDecl(fp, od, FALSE); + prcode(fp, ";\n"); + } + + prcode(fp, +"\n" +"public:\n" +" sipWrapper *sipPySelf;\n" + ); + + /* The private declarations. */ + + prcode(fp, +"\n" +"private:\n" +" sip%C(const sip%C &);\n" +" sip%C &operator = (const sip%C &);\n" + ,classFQCName(cd),classFQCName(cd) + ,classFQCName(cd),classFQCName(cd)); + + if ((nrVirts = countVirtuals(cd)) > 0) + prcode(fp, +"\n" +" sipMethodCache sipPyMethods[%d];\n" + ,nrVirts); + + prcode(fp, +"};\n" + ); +} + + +/* + * Generate the C++ declaration for an overload. + */ +void prOverloadDecl(FILE *fp, overDef *od, int defval) +{ + int a; + + normaliseArgs(od->cppsig); + + generateBaseType(&od->cppsig->result, fp); + + prcode(fp, " %O(", od); + + for (a = 0; a < od->cppsig->nrArgs; ++a) + { + argDef *ad = &od->cppsig->args[a]; + + if (a > 0) + prcode(fp, ","); + + generateBaseType(ad, fp); + + if (defval && ad->defval != NULL) + { + prcode(fp, " = "); + generateExpression(ad->defval, fp); + } + } + + prcode(fp, ")%s%X", (isConst(od) ? " const" : ""), od->exceptions); + + restoreArgs(od->cppsig); +} + + +/* + * Generate typed arguments. + */ +static void generateArgs(signatureDef *sd,funcArgType ftype,FILE *fp) +{ + int a; + + for (a = 0; a < sd->nrArgs; ++a) + { + if (a > 0) + prcode(fp,","); + + generateSingleArg(&sd->args[a],a,ftype,fp); + } +} + + +/* + * Generate the declaration of a named variable to hold a result from a C++ + * function call. + */ +static void generateNamedValueType(argDef *ad,char *name,FILE *fp) +{ + argDef mod = *ad; + + if (ad->nrderefs == 0) + if (ad->atype == class_type || ad->atype == mapped_type) + mod.nrderefs = 1; + else + resetIsConstArg(&mod); + + resetIsReference(&mod); + generateNamedBaseType(&mod,name,fp); +} + + +/* + * Generate a single argument. + */ +static void generateSingleArg(argDef *ad,int argnr,funcArgType ftype,FILE *fp) +{ + char name[50]; + int genType, genName, derefPtr; + + /* Break the type down to individual modifications. */ + + genType = FALSE; + genName = FALSE; + derefPtr = FALSE; + + switch (ftype) + { + case Call: + genName = TRUE; + derefPtr = TRUE; + break; + + case Declaration: + genType = TRUE; + break; + + case Definition: + genType = TRUE; + genName = TRUE; + break; + } + + if (genName) + { + char *ind = ""; + + if (derefPtr) + switch (ad->atype) + { + case sstring_type: + case ustring_type: + case string_type: + case wstring_type: + if (ad->nrderefs > (isOutArg(ad) ? 0 : 1)) + ind = "&"; + + break; + + case mapped_type: + case class_type: + if (ad->nrderefs == 2) + ind = "&"; + else if (ad->nrderefs == 0) + ind = "*"; + + break; + + case struct_type: + case void_type: + if (ad->nrderefs == 2) + ind = "&"; + + break; + + default: + if (ad->nrderefs == 1) + ind = "&"; + } + + sprintf(name,"%sa%d",ind,argnr); + } + else + name[0] = '\0'; + + if (genType) + generateNamedBaseType(ad,name,fp); + else if (genName) + prcode(fp,name); +} + + +/* + * Generate a C++ type. + */ +static void generateBaseType(argDef *ad,FILE *fp) +{ + generateNamedBaseType(ad,"",fp); +} + + +/* + * Generate a C++ type and name. + */ +static void generateNamedBaseType(argDef *ad,char *name,FILE *fp) +{ + int nr_derefs = ad->nrderefs; + + /* + * A function type is handled differently because of the position of + * the name. + */ + if (ad->atype == function_type) + { + int i; + signatureDef *sig = ad->u.sa; + + generateBaseType(&sig->result,fp); + + prcode(fp," ("); + + for (i = 0; i < nr_derefs; ++i) + prcode(fp,"*"); + + prcode(fp,"%s)(",name); + generateArgs(sig,Declaration,fp); + prcode(fp,")"); + + return; + } + + if (isConstArg(ad)) + prcode(fp,"const "); + + switch (ad->atype) + { + case sstring_type: + prcode(fp,"signed char"); + break; + + case ustring_type: + prcode(fp,"unsigned char"); + break; + + case wstring_type: + prcode(fp,"wchar_t"); + break; + + case signal_type: + case slot_type: + case anyslot_type: + case slotcon_type: + case slotdis_type: + nr_derefs = 1; + + /* Drop through. */ + + case string_type: + prcode(fp,"char"); + break; + + case ushort_type: + prcode(fp,"unsigned short"); + break; + + case short_type: + prcode(fp,"short"); + break; + + case uint_type: + prcode(fp,"unsigned"); + break; + + case int_type: + case cint_type: + prcode(fp,"int"); + break; + + case ulong_type: + prcode(fp,"unsigned long"); + break; + + case long_type: + prcode(fp,"long"); + break; + + case ulonglong_type: + prcode(fp,"unsigned PY_LONG_LONG"); + break; + + case longlong_type: + prcode(fp,"PY_LONG_LONG"); + break; + + case struct_type: + prcode(fp,"struct %S",ad->u.sname); + break; + + case void_type: + prcode(fp,"void"); + break; + + case bool_type: + case cbool_type: + prcode(fp,"bool"); + break; + + case float_type: + case cfloat_type: + prcode(fp,"float"); + break; + + case double_type: + case cdouble_type: + prcode(fp,"double"); + break; + + case defined_type: + /* + * The only defined types still remaining are arguments to + * templates. + */ + + prcode(fp,"%S",ad->u.snd); + break; + + case rxcon_type: + case rxdis_type: + nr_derefs = 1; + prcode(fp,"QObject"); + break; + + case mapped_type: + generateBaseType(&ad->u.mtd->type,fp); + break; + + case class_type: + prcode(fp,"%U",ad->u.cd); + break; + + case template_type: + { + static const char tail[] = ">"; + int a; + templateDef *td = ad->u.td; + + prcode(fp, "%S%s", td->fqname, (prcode_xml ? "<" : "<")); + + for (a = 0; a < td->types.nrArgs; ++a) + { + if (a > 0) + prcode(fp,","); + + generateBaseType(&td->types.args[a],fp); + } + + if (prcode_last == tail) + prcode(fp, " "); + + prcode(fp, (prcode_xml ? ">" : tail)); + break; + } + + case enum_type: + prcode(fp,"%E",ad->u.ed); + break; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + case qobject_type: + case ellipsis_type: + prcode(fp, "PyObject *"); + break; + } + + if (nr_derefs > 0) + { + int i; + + prcode(fp," "); + + for (i = 0; i < nr_derefs; ++i) + prcode(fp,"*"); + } + + if (isReference(ad)) + prcode(fp, (prcode_xml ? "&" : "&")); + + if (*name != '\0') + { + if (nr_derefs == 0) + prcode(fp," "); + + prcode(fp,name); + } +} + + +/* + * Generate the definition of an argument variable and any supporting + * variables. + */ +static void generateVariable(argDef *ad,int argnr,FILE *fp) +{ + argType atype = ad->atype; + argDef orig; + + if (isInArg(ad) && ad->defval != NULL && + (atype == class_type || atype == mapped_type) && + (ad->nrderefs == 0 || isReference(ad))) + { + /* + * Generate something to hold the default value as it cannot be + * assigned straight away. + */ + prcode(fp, +" %B a%ddef = ",ad,argnr); + + generateExpression(ad->defval,fp); + + prcode(fp,";\n" + ); + } + + /* Adjust the type so we have the type that will really handle it. */ + + orig = *ad; + + switch (atype) + { + case sstring_type: + case ustring_type: + case string_type: + case wstring_type: + if (!isReference(ad)) + if (ad->nrderefs == 2) + ad->nrderefs = 1; + else if (ad->nrderefs == 1 && isOutArg(ad)) + ad->nrderefs = 0; + + break; + + case mapped_type: + case class_type: + case void_type: + case struct_type: + ad->nrderefs = 1; + break; + + default: + ad->nrderefs = 0; + } + + /* Array sizes are always integers. */ + if (isArraySize(ad)) + ad->atype = int_type; + + resetIsReference(ad); + + if (ad->nrderefs == 0) + resetIsConstArg(ad); + + prcode(fp, +" %B a%d",ad,argnr); + + if (atype == anyslot_type) + prcode(fp, "Name"); + + *ad = orig; + + generateDefaultValue(ad, argnr, fp); + + prcode(fp,";\n" + ); + + /* Some types have supporting variables. */ + if (isInArg(ad)) + switch (atype) + { + case class_type: + if (ad->u.cd->convtocode != NULL && !isConstrained(ad)) + prcode(fp, +" int a%dState = 0;\n" + ,argnr); + + if (isGetWrapper(ad)) + prcode(fp, +" PyObject *a%dWrapper;\n" + ,argnr); + + break; + + case mapped_type: + prcode(fp, +" int a%dState = 0;\n" + ,argnr); + break; + + case anyslot_type: + prcode(fp, +" PyObject *a%dCallable", argnr); + generateDefaultValue(ad, argnr, fp); + prcode(fp, ";\n" + ); + break; + } +} + + +/* + * Generate a default value. + */ +static void generateDefaultValue(argDef *ad, int argnr, FILE *fp) +{ + if (isInArg(ad) && ad->defval != NULL) + { + prcode(fp," = "); + + if ((ad->atype == class_type || ad->atype == mapped_type) && + (ad->nrderefs == 0 || isReference(ad))) + prcode(fp, "&a%ddef", argnr); + else + generateExpression(ad->defval,fp); + } +} + + +/* + * Generate a simple function call. + */ +static void generateSimpleFunctionCall(fcallDef *fcd,FILE *fp) +{ + int i; + + prcode(fp, "%B(", &fcd->type); + + for (i = 0; i < fcd->nrArgs; ++i) + { + if (i > 0) + prcode(fp,","); + + generateExpression(fcd->args[i],fp); + } + + prcode(fp,")"); +} + + +/* + * Generate the type structure that contains all the information needed by the + * metatype. A sub-set of this is used to extend namespaces. + */ +static void generateTypeDefinition(sipSpec *pt, classDef *cd, FILE *fp) +{ + char *mname = pt->module->name; + const char *sep; + int is_slots, nr_methods, nr_enums; + int is_inst_class, is_inst_voidp, is_inst_char, is_inst_string; + int is_inst_int, is_inst_long, is_inst_ulong, is_inst_longlong; + int is_inst_ulonglong, is_inst_double, is_inst_enum; + memberDef *md; + + if (cd->supers != NULL) + { + classList *cl; + + prcode(fp, +"\n" +"\n" +"/* Define this type's super-types. */\n" +"static sipEncodedClassDef supers_%C[] = {",classFQCName(cd)); + + for (cl = cd->supers; cl != NULL; cl = cl->next) + { + if (cl != cd->supers) + prcode(fp,", "); + + generateEncodedClass(pt,cl->cd,(cl->next == NULL),fp); + } + + prcode(fp,"};\n" + ); + } + + /* Generate the slots table. */ + is_slots = FALSE; + + for (md = cd->members; md != NULL; md = md->next) + { + const char *stype; + + if (md->slot == no_slot) + continue; + + if (!is_slots) + { + prcode(fp, +"\n" +"\n" +"/* Define this type's Python slots. */\n" +"static sipPySlotDef slots_%C[] = {\n" + ,classFQCName(cd)); + + is_slots = TRUE; + } + + if ((stype = slotName(md->slot)) != NULL) + prcode(fp, +" {(void *)slot_%C_%s, %s},\n" + ,classFQCName(cd),md->pyname->text,stype); + } + + if (is_slots) + prcode(fp, +" {0, (sipPySlotType)0}\n" +"};\n" + ); + + /* Generate the attributes tables. */ + nr_methods = generateMethodTable(cd,fp); + nr_enums = generateEnumMemberTable(pt,cd,fp); + + /* Generate each instance table. */ + is_inst_class = generateClasses(pt,cd,fp); + is_inst_voidp = generateVoidPointers(pt,cd,fp); + is_inst_char = generateChars(pt,cd,fp); + is_inst_string = generateStrings(pt,cd,fp); + is_inst_int = generateInts(pt,cd,fp); + is_inst_long = generateLongs(pt,cd,fp); + is_inst_ulong = generateUnsignedLongs(pt,cd,fp); + is_inst_longlong = generateLongLongs(pt,cd,fp); + is_inst_ulonglong = generateUnsignedLongLongs(pt,cd,fp); + is_inst_double = generateDoubles(pt,cd,fp); + is_inst_enum = generateEnums(pt,cd,fp); + + prcode(fp, +"\n" +"\n" +"sipTypeDef sipType_%s_%C = {\n" +" 0,\n" +" ", mname, classFQCName(cd)); + + sep = ""; + + if (cd->userflags) + { + prcode(fp, "%s%x", sep, ((cd->userflags << TYPE_FLAGS_SHIFT) & TYPE_FLAGS_MASK)); + sep = "|"; + } + + if (isAbstractClass(cd)) + { + prcode(fp, "%sSIP_TYPE_ABSTRACT", sep); + sep = "|"; + } + + if (cd->subbase != NULL) + { + prcode(fp, "%sSIP_TYPE_SCC", sep); + sep = "|"; + } + + if (*sep == '\0') + prcode(fp, "0"); + + prcode(fp, ",\n"); + + if (cd->real != NULL) + prcode(fp, +" 0,\n" + ); + else if (cd->ecd != NULL && cd->ecd->real != NULL) + prcode(fp, +" \"%s.%P\",\n" + , cd->ecd->real->iff->module->name, cd->ecd, cd->pyname); + else + prcode(fp, +" \"%s.%P\",\n" + , mname, cd->ecd, cd->pyname); + + if (isRenamedClass(cd)) + prcode(fp, +" \"%S\",\n" + , classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + prcode(fp, +" "); + + if (cd->real != NULL) + generateEncodedClass(pt, cd->real, 0, fp); + else if (cd->ecd != NULL) + generateEncodedClass(pt, cd->ecd, 0, fp); + else + prcode(fp, "{0, 0, 1}"); + + prcode(fp, ",\n" + ); + + if (cd->supers != NULL) + prcode(fp, +" supers_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (is_slots) + prcode(fp, +" slots_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (nr_methods == 0) + prcode(fp, +" 0, 0,\n" + ); + else + prcode(fp, +" %d, methods_%C,\n" + ,nr_methods,classFQCName(cd)); + + if (nr_enums == 0) + prcode(fp, +" 0, 0,\n" + ); + else + prcode(fp, +" %d, enummembers_%C,\n" + ,nr_enums,classFQCName(cd)); + + if (hasVarHandlers(cd)) + prcode(fp, +" variables_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (canCreate(cd)) + prcode(fp, +" init_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->travcode != NULL) + prcode(fp, +" traverse_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->clearcode != NULL) + prcode(fp, +" clear_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->readbufcode != NULL) + prcode(fp, +" getreadbuffer_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->writebufcode != NULL) + prcode(fp, +" getwritebuffer_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->segcountcode != NULL) + prcode(fp, +" getsegcount_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->charbufcode != NULL) + prcode(fp, +" getcharbuffer_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (needDealloc(cd)) + prcode(fp, +" dealloc_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + if (cd->iff->type == namespace_iface || generating_c) + prcode(fp, +" 0,\n" +" 0,\n" + ); + else + prcode(fp, +" cast_%C,\n" +" release_%C,\n" + , classFQCName(cd) + , classFQCName(cd)); + + if (cd->iff->type == namespace_iface) + prcode(fp, +" 0,\n" +" 0,\n" + ); + else + { + prcode(fp, +" forceConvertTo_%C,\n" + ,classFQCName(cd)); + + if (cd->convtocode != NULL) + prcode(fp, +" convertTo_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + } + + if (!optNoEmitters(pt) && hasSigSlots(cd)) + prcode(fp, +" signals_%C,\n" + ,classFQCName(cd)); + else + prcode(fp, +" 0,\n" + ); + + prcode(fp, +" {"); + + if (is_inst_class) + prcode(fp,"classInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_voidp) + prcode(fp,"voidPtrInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_char) + prcode(fp,"charInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_string) + prcode(fp,"stringInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_int) + prcode(fp,"intInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_long) + prcode(fp,"longInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_ulong) + prcode(fp,"unsignedLongInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_longlong) + prcode(fp,"longLongInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_ulonglong) + prcode(fp,"unsignedLongLongInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_double) + prcode(fp,"doubleInstances_%C, ",classFQCName(cd)); + else + prcode(fp,"0, "); + + if (is_inst_enum) + prcode(fp,"enumInstances_%C",classFQCName(cd)); + else + prcode(fp,"0"); + + prcode(fp,"},\n" +" 0\n" +"};\n" + ); +} + + +/* + * Return the sip module's string equivalent of a slot. + */ +static const char *slotName(slotType st) +{ + const char *sn; + + switch (st) + { + case str_slot: + sn = "str_slot"; + break; + + case int_slot: + sn = "int_slot"; + break; + + case long_slot: + sn = "long_slot"; + break; + + case float_slot: + sn = "float_slot"; + break; + + case len_slot: + sn = "len_slot"; + break; + + case contains_slot: + sn = "contains_slot"; + break; + + case add_slot: + sn = "add_slot"; + break; + + case concat_slot: + sn = "concat_slot"; + break; + + case sub_slot: + sn = "sub_slot"; + break; + + case mul_slot: + sn = "mul_slot"; + break; + + case repeat_slot: + sn = "repeat_slot"; + break; + + case div_slot: + sn = "div_slot"; + break; + + case mod_slot: + sn = "mod_slot"; + break; + + case and_slot: + sn = "and_slot"; + break; + + case or_slot: + sn = "or_slot"; + break; + + case xor_slot: + sn = "xor_slot"; + break; + + case lshift_slot: + sn = "lshift_slot"; + break; + + case rshift_slot: + sn = "rshift_slot"; + break; + + case iadd_slot: + sn = "iadd_slot"; + break; + + case iconcat_slot: + sn = "iconcat_slot"; + break; + + case isub_slot: + sn = "isub_slot"; + break; + + case imul_slot: + sn = "imul_slot"; + break; + + case irepeat_slot: + sn = "irepeat_slot"; + break; + + case idiv_slot: + sn = "idiv_slot"; + break; + + case imod_slot: + sn = "imod_slot"; + break; + + case iand_slot: + sn = "iand_slot"; + break; + + case ior_slot: + sn = "ior_slot"; + break; + + case ixor_slot: + sn = "ixor_slot"; + break; + + case ilshift_slot: + sn = "ilshift_slot"; + break; + + case irshift_slot: + sn = "irshift_slot"; + break; + + case invert_slot: + sn = "invert_slot"; + break; + + case call_slot: + sn = "call_slot"; + break; + + case getitem_slot: + sn = "getitem_slot"; + break; + + case setitem_slot: + sn = "setitem_slot"; + break; + + case delitem_slot: + sn = "delitem_slot"; + break; + + case lt_slot: + sn = "lt_slot"; + break; + + case le_slot: + sn = "le_slot"; + break; + + case eq_slot: + sn = "eq_slot"; + break; + + case ne_slot: + sn = "ne_slot"; + break; + + case gt_slot: + sn = "gt_slot"; + break; + + case ge_slot: + sn = "ge_slot"; + break; + + case cmp_slot: + sn = "cmp_slot"; + break; + + case nonzero_slot: + sn = "nonzero_slot"; + break; + + case neg_slot: + sn = "neg_slot"; + break; + + case pos_slot: + sn = "pos_slot"; + break; + + case abs_slot: + sn = "abs_slot"; + break; + + case repr_slot: + sn = "repr_slot"; + break; + + case hash_slot: + sn = "hash_slot"; + break; + + default: + sn = NULL; + } + + return sn; +} + + +/* + * Generate the code to register a class as a Qt metatype. + */ +static void generateRegisterMetaType(classDef *cd, FILE *fp) +{ + int pub_def_ctor, pub_copy_ctor; + ctorDef *ct; + + /* + * We register types with Qt if the class is not abstract, has a public + * default ctor, a public copy ctor, a public dtor and isn't one of the + * internally supported types. + */ + if (isAbstractClass(cd)) + return; + + if (!isPublicDtor(cd)) + return; + + if (classFQCName(cd)->next == NULL) + { + if (strcmp(classBaseName(cd), "QChar") == 0) + return; + + if (strcmp(classBaseName(cd), "QString") == 0) + return; + + if (strcmp(classBaseName(cd), "QByteArray") == 0) + return; + } + + pub_def_ctor = pub_copy_ctor = FALSE; + + for (ct = cd->ctors; ct != NULL; ct = ct->next) + { + if (ct->cppsig == NULL || !isPublicCtor(ct)) + continue; + + if (ct->cppsig->nrArgs == 0) + pub_def_ctor = TRUE; + else if (ct->cppsig->nrArgs == 1) + { + argDef *ad = &ct->cppsig->args[0]; + + if (ad->atype == class_type && ad->u.cd == cd && isReference(ad) && + isConstArg(ad) && ad->nrderefs == 0 && ad->defval == NULL) + pub_copy_ctor = TRUE; + } + } + + if (pub_def_ctor && pub_copy_ctor) + prcode(fp, +" qRegisterMetaType<%S>(\"%S\");\n" + , classFQCName(cd), classFQCName(cd)); +} + + +/* + * Generate the initialisation function or cast operators for the type. + */ +static void generateTypeInit(sipSpec *pt, classDef *cd, FILE *fp) +{ + ctorDef *ct; + int need_self, need_owner; + + /* + * See if we need to name the self and owner arguments so that we can + * avoid a compiler warning about an unused argument. + */ + need_self = (generating_c || hasShadow(cd)); + need_owner = generating_c; + + for (ct = cd->ctors; ct != NULL; ct = ct->next) + { + int a; + + if (usedInCode(ct->methodcode, "sipSelf")) + need_self = TRUE; + + for (a = 0; a < ct->pysig.nrArgs; ++a) + if (isThisTransferred(&ct->pysig.args[a])) + { + need_owner = TRUE; + break; + } + } + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static void *init_%C(sipWrapper *, PyObject *, sipWrapper **, int *);}\n" + , classFQCName(cd)); + + prcode(fp, +"static void *init_%C(sipWrapper *%s,PyObject *sipArgs,sipWrapper **%s,int *sipArgsParsed)\n" +"{\n" + ,classFQCName(cd),(need_self ? "sipSelf" : ""),(need_owner ? "sipOwner" : "")); + + if (hasShadow(cd)) + prcode(fp, +" sip%C *sipCpp = 0;\n" + ,classFQCName(cd)); + else + prcode(fp, +" %U *sipCpp = 0;\n" + ,cd); + + if (tracing) + prcode(fp, +"\n" +" sipTrace(SIP_TRACE_INITS,\"init_%C()\\n\");\n" + ,classFQCName(cd)); + + /* + * Generate the code that parses the Python arguments and calls the + * correct constructor. + */ + for (ct = cd->ctors; ct != NULL; ct = ct->next) + { + int needSecCall, error_flag = FALSE; + + if (isPrivateCtor(ct)) + continue; + + prcode(fp, +"\n" +" if (!sipCpp)\n" +" {\n" + ); + + if (ct->methodcode != NULL && needErrorFlag(ct->methodcode)) + { + prcode(fp, +" int sipIsErr = 0;\n" + ); + + error_flag = TRUE; + } + + needSecCall = generateArgParser(pt, &ct->pysig, cd, ct, NULL, FALSE, fp); + generateConstructorCall(cd,ct,error_flag,fp); + + if (needSecCall) + { + prcode(fp, +" }\n" +"\n" +" if (!sipCpp)\n" +" {\n" + ); + + if (error_flag) + prcode(fp, +" int sipIsErr = 0;\n" + ); + + generateArgParser(pt, &ct->pysig, cd, ct, NULL, TRUE, fp); + generateConstructorCall(cd,ct,error_flag,fp); + } + + prcode(fp, +" }\n" + ); + } + + if (hasShadow(cd)) + prcode(fp, +"\n" +" if (sipCpp)\n" +" sipCpp->sipPySelf = sipSelf;\n" + ); + + prcode(fp, +"\n" +" return sipCpp;\n" +"}\n" + ); +} + + +/* + * Count the number of virtual members in a class. + */ +static int countVirtuals(classDef *cd) +{ + int nrvirts; + virtOverDef *vod; + + nrvirts = 0; + + for (vod = cd->vmembers; vod != NULL; vod = vod->next) + if (!isPrivate(&vod->o)) + ++nrvirts; + + return nrvirts; +} + + +/* + * Generate the try block for a call. + */ +static void generateTry(throwArgs *ta,FILE *fp) +{ + /* + * Generate the block if there was no throw specifier, or a non-empty + * throw specifier. + */ + if (exceptions && (ta == NULL || ta->nrArgs > 0)) + prcode(fp, +" try\n" +" {\n" + ); +} + + +/* + * Generate the catch block for a call. + */ +static void generateCatch(throwArgs *ta, signatureDef *sd, FILE *fp) +{ + /* + * Generate the block if there was no throw specifier, or a non-empty + * throw specifier. + */ + if (exceptions && (ta == NULL || ta->nrArgs > 0)) + { + prcode(fp, +" }\n" + ); + + if (ta == NULL) + { + prcode(fp, +" catch (...)\n" +" {\n" + ); + + deleteTemps(sd, fp); + + if (release_gil) + prcode(fp, +" Py_BLOCK_THREADS\n" +"\n" + ); + + prcode(fp, +" sipRaiseUnknownException();\n" +" return NULL;\n" +" }\n" + ); + } + else + { + int a; + + for (a = 0; a < ta->nrArgs; ++a) + { + exceptionDef *xd = ta->args[a]; + scopedNameDef *ename = xd->iff->fqcname; + + prcode(fp, +" catch (%S &%s)\n" +" {\n" + ,ename,(xd->cd != NULL || usedInCode(xd->raisecode, "sipExceptionRef")) ? "sipExceptionRef" : ""); + + deleteTemps(sd, fp); + + if (xd->cd != NULL) + { + /* The exception is a wrapped class. */ + + prcode(fp, +" /* Hope that there is a valid copy ctor. */\n" +" %S *sipExceptionCopy = new %S(sipExceptionRef);\n" +"\n" + ,ename,ename); + + if (release_gil) + prcode(fp, +" Py_BLOCK_THREADS\n" +"\n" + ); + + prcode(fp, +" sipRaise%sClassException(sipClass_%C,sipExceptionCopy);\n" + ,(xd->cd->subbase != NULL ? "Sub" : ""),ename); + } + else + generateCppCodeBlock(xd->raisecode,fp); + + prcode(fp, +"\n" +" return NULL;\n" +" }\n" + ); + } + } + } +} + + +/* + * Generate a throw specifier. + */ +static void generateThrowSpecifier(throwArgs *ta,FILE *fp) +{ + if (exceptions && ta != NULL) + { + int a; + + prcode(fp," throw("); + + for (a = 0; a < ta->nrArgs; ++a) + { + if (a > 0) + prcode(fp,","); + + prcode(fp,"%S",ta->args[a]->iff->fqcname); + } + + prcode(fp,")"); + } +} + + +/* + * Generate a single constructor call. + */ +static void generateConstructorCall(classDef *cd,ctorDef *ct,int error_flag, + FILE *fp) +{ + prcode(fp, +" {\n" + ); + + /* Call any pre-hook. */ + if (ct->prehook != NULL) + prcode(fp, +" sipCallHook(\"%s\");\n" +"\n" + ,ct->prehook); + + if (ct->methodcode != NULL) + generateCppCodeBlock(ct->methodcode,fp); + else if (generating_c) + prcode(fp, +" sipCpp = sipMalloc(sizeof (%S));\n" + ,classFQCName(cd)); + else + { + int rgil = ((release_gil || isReleaseGILCtor(ct)) && !isHoldGILCtor(ct)); + + if (rgil) + prcode(fp, +" Py_BEGIN_ALLOW_THREADS\n" + ); + + generateTry(ct->exceptions,fp); + + if (hasShadow(cd)) + prcode(fp, +" sipCpp = new sip%C(",classFQCName(cd)); + else + prcode(fp, +" sipCpp = new %U(",cd); + + if (isCastCtor(ct)) + { + classDef *ocd; + + /* We have to fiddle the type to generate the correct code. */ + ocd = ct->pysig.args[0].u.cd; + ct->pysig.args[0].u.cd = cd; + prcode(fp, "a0->operator %B()", &ct->pysig.args[0]); + ct->pysig.args[0].u.cd = ocd; + } + else + generateArgs(ct->cppsig, Call, fp); + + prcode(fp,");\n" + ); + + generateCatch(ct->exceptions, &ct->pysig, fp); + + if (rgil) + prcode(fp, +" Py_END_ALLOW_THREADS\n" + ); + } + + gc_ellipsis(&ct->pysig, fp); + + deleteTemps(&ct->pysig, fp); + + if (error_flag) + prcode(fp, +"\n" +" if (sipIsErr)\n" +" return 0;\n" + ); + + /* Call any post-hook. */ + if (ct->posthook != NULL) + prcode(fp, +"\n" +" sipCallHook(\"%s\");\n" + ,ct->posthook); + + prcode(fp, +" }\n" + ); +} + + +/* + * See if a member overload should be skipped. + */ +static int skipOverload(overDef *od,memberDef *md,classDef *cd,classDef *ccd, + int want_local) +{ + /* Skip if it's not the right name. */ + if (od->common != md) + return TRUE; + + /* Skip if it's a signal. */ + if (isSignal(od)) + return TRUE; + + /* Skip if it's a private abstract. */ + if (isAbstract(od) && isPrivate(od)) + return TRUE; + + /* + * If we are disallowing them, skip if it's not in the current class + * unless it is protected. + */ + if (want_local && !isProtected(od) && ccd != cd) + return TRUE; + + return FALSE; +} + + +/* + * Generate a class member function. + */ +static void generateFunction(sipSpec *pt,memberDef *md,overDef *overs, + classDef *cd,classDef *ocd,FILE *fp) +{ + overDef *od; + int need_method, need_self, need_args, need_selfarg; + + /* + * Check that there is at least one overload that needs to be handled. + * See if we can avoid naming the "self" argument (and suppress a + * compiler warning). Finally see if we need to remember if "self" was + * explicitly passed as an argument. + */ + need_method = need_self = need_args = need_selfarg = FALSE; + + for (od = overs; od != NULL; od = od->next) + { + /* + * Skip protected methods if we don't have the means to handle + * them. + */ + if (isProtected(od) && !hasShadow(cd)) + continue; + + if (!skipOverload(od,md,cd,ocd,TRUE)) + { + need_method = TRUE; + + if (!isPrivate(od)) + { + need_args = TRUE; + + if (!isStatic(od)) + { + need_self = TRUE; + + if (isAbstract(od) || isVirtual(od) || isVirtualReimp(od) || usedInCode(od->methodcode, "sipSelfWasArg")) + need_selfarg = TRUE; + } + } + } + } + + if (need_method) + { + char *pname = md->pyname->text; + + prcode(fp, +"\n" +"\n" + ); + + if (!generating_c) + prcode(fp, +"extern \"C\" {static PyObject *meth_%C_%s(PyObject *, PyObject *);}\n" + , classFQCName(cd), pname); + + prcode(fp, +"static PyObject *meth_%C_%s(PyObject *%s,PyObject *%s)\n" +"{\n" + ,classFQCName(cd),pname,(need_self ? "sipSelf" : ""),(need_args ? "sipArgs" : "")); + + if (tracing) + prcode(fp, +" sipTrace(SIP_TRACE_METHODS,\"meth_%C_%s()\\n\");\n" +"\n" + ,classFQCName(cd),pname); + + if (need_args) + prcode(fp, +" int sipArgsParsed = 0;\n" + ); + + if (need_selfarg) + prcode(fp, +" bool sipSelfWasArg = !sipSelf;\n" + ); + + for (od = overs; od != NULL; od = od->next) + { + /* + * If we are handling one variant then we must handle + * them all. + */ + if (skipOverload(od,md,cd,ocd,FALSE)) + continue; + + if (isPrivate(od)) + continue; + + generateFunctionBody(pt,od,cd,ocd,TRUE,fp); + } + + prcode(fp, +"\n" +" /* Raise an exception if the arguments couldn't be parsed. */\n" +" sipNoMethod(%s,%N,%N);\n" +"\n" +" return NULL;\n" +"}\n" + ,(need_args ? "sipArgsParsed" : "0"),cd->iff->name,md->pyname); + } +} + + +/* + * Generate the function calls for a particular overload. + */ +static void generateFunctionBody(sipSpec *pt,overDef *od,classDef *cd, + classDef *ocd,int deref,FILE *fp) +{ + int needSecCall; + signatureDef saved; + + prcode(fp, +"\n" +" {\n" + ); + + /* In case we have to fiddle with it. */ + saved = od->pysig; + + if (isNumberSlot(od->common)) + { + /* + * Number slots must have two arguments because we parse them + * slightly differently. + */ + if (od->pysig.nrArgs == 1) + { + od->pysig.nrArgs = 2; + od->pysig.args[1] = od->pysig.args[0]; + + /* Insert self as the first argument. */ + od->pysig.args[0].atype = class_type; + od->pysig.args[0].name = NULL; + od->pysig.args[0].argflags = ARG_IS_REF|ARG_IN; + od->pysig.args[0].nrderefs = 0; + od->pysig.args[0].defval = NULL; + od->pysig.args[0].u.cd = ocd; + } + + generateArgParser(pt, &od->pysig, cd, NULL, od, FALSE, fp); + needSecCall = FALSE; + } + else if (isIntArgSlot(od->common) || isZeroArgSlot(od->common)) + needSecCall = FALSE; + else + needSecCall = generateArgParser(pt, &od->pysig, cd, NULL, od, FALSE, fp); + + generateFunctionCall(cd,ocd,od,deref,fp); + + if (needSecCall) + { + prcode(fp, +" }\n" +"\n" +" {\n" + ); + + generateArgParser(pt, &od->pysig, cd, NULL, od, TRUE, fp); + generateFunctionCall(cd,ocd,od,deref,fp); + } + + prcode(fp, +" }\n" + ); + + od->pysig = saved; +} + + +/* + * Generate the code to handle the result of a call to a member function. + */ +static void generateHandleResult(overDef *od,int isNew,char *prefix,FILE *fp) +{ + char *vname, vnamebuf[50]; + int a, nrvals, only, has_owner; + argDef *res, *ad; + + res = &od->pysig.result; + + if (res->atype == void_type && res->nrderefs == 0) + res = NULL; + + /* See if we are returning 0, 1 or more values. */ + nrvals = 0; + + if (res != NULL) + { + only = -1; + ++nrvals; + } + + has_owner = FALSE; + + for (a = 0; a < od->pysig.nrArgs; ++a) + { + if (isOutArg(&od->pysig.args[a])) + { + only = a; + ++nrvals; + } + + if (isThisTransferred(&od->pysig.args[a])) + has_owner = TRUE; + } + + /* Handle the trivial case. */ + if (nrvals == 0) + { + prcode(fp, +" Py_INCREF(Py_None);\n" +" %s Py_None;\n" + ,prefix); + + return; + } + + /* Handle results that are classes or mapped types separately. */ + if (res != NULL) + if (res->atype == mapped_type) + { + prcode(fp, +" PyObject *sipResObj = sipConvertFromMappedType("); + + if (isConstArg(res)) + prcode(fp,"const_cast<%b *>(sipRes)",res); + else + prcode(fp,"sipRes"); + + prcode(fp,",sipMappedType_%T,%s);\n" + , res, isResultTransferredBack(od) ? "Py_None" : "NULL"); + + if (isNew) + prcode(fp, +" delete sipRes;\n" + ); + + /* Shortcut if this is the only value returned. */ + if (nrvals == 1) + { + prcode(fp, +"\n" +" %s sipResObj;\n" + ,prefix); + + return; + } + } + else if (res->atype == class_type) + { + classDef *cd = res->u.cd; + + if (isNew || isFactory(od)) + { + prcode(fp, +" %s sipConvertFromNewInstance(",(nrvals == 1 ? prefix : "PyObject *sipResObj =")); + + if (isConstArg(res)) + prcode(fp,"const_cast<%b *>(sipRes)",res); + else + prcode(fp,"sipRes"); + + prcode(fp,",sipClass_%C,%s);\n" + ,classFQCName(cd),((has_owner && isFactory(od)) ? "(PyObject *)sipOwner" : "NULL")); + + /* + * Shortcut if this is the only value returned. + */ + if (nrvals == 1) + return; + } + else + { + prcode(fp, +" %s sipConvertFromInstance(",(nrvals == 1 ? prefix : "PyObject *sipResObj =")); + + if (isConstArg(res)) + prcode(fp,"const_cast<%b *>(sipRes)",res); + else + prcode(fp,"sipRes"); + + prcode(fp, ",sipClass_%C,%s);\n" + , classFQCName(cd), (isResultTransferredBack(od) ? "Py_None" : "NULL")); + + /* + * Shortcut if this is the only value returned. + */ + if (nrvals == 1) + return; + } + } + + /* If there are multiple values then build a tuple. */ + if (nrvals > 1) + { + prcode(fp, +" %s sipBuildResult(0,\"(",prefix); + + /* Build the format string. */ + if (res != NULL) + prcode(fp,"%c",((res->atype == mapped_type || res->atype == class_type) ? 'R' : getBuildResultFormat(res))); + + for (a = 0; a < od->pysig.nrArgs; ++a) + { + argDef *ad = &od->pysig.args[a]; + + if (isOutArg(ad)) + prcode(fp,"%c",getBuildResultFormat(ad)); + } + + prcode(fp,")\""); + + /* Pass the values for conversion. */ + if (res != NULL) + { + prcode(fp,",sipRes"); + + if (res->atype == mapped_type || res->atype == class_type) + prcode(fp,"Obj"); + else if (res->atype == enum_type && res->u.ed->fqcname != NULL) + prcode(fp,",sipEnum_%C",res->u.ed->fqcname); + } + + for (a = 0; a < od->pysig.nrArgs; ++a) + { + argDef *ad = &od->pysig.args[a]; + + if (isOutArg(ad)) + { + prcode(fp,",a%d",a); + + if (ad->atype == mapped_type) + prcode(fp, ",sipMappedType_%T,%s", ad, (isTransferredBack(ad) ? "Py_None" : "NULL")); + else if (ad->atype == class_type) + prcode(fp, ",sipClass_%C,%s", classFQCName(ad->u.cd), (isTransferredBack(ad) ? "Py_None" : "NULL")); + else if (ad->atype == enum_type && ad->u.ed->fqcname != NULL) + prcode(fp,",sipEnum_%C",ad->u.ed->fqcname); + } + } + + prcode(fp,");\n" + ); + + /* All done for multiple values. */ + return; + } + + /* Deal with the only returned value. */ + if (only < 0) + { + ad = res; + vname = "sipRes"; + } + else + { + ad = &od->pysig.args[only]; + + sprintf(vnamebuf,"a%d",only); + vname = vnamebuf; + } + + switch (ad->atype) + { + case mapped_type: + prcode(fp, +" %s sipConvertFromMappedType(", prefix); + + if (isConstArg(ad)) + prcode(fp,"const_cast<%b *>(%s)",ad,vname); + else + prcode(fp,"%s",vname); + + prcode(fp,",sipMappedType_%T,%s);\n" + , ad, (isTransferredBack(ad) ? "Py_None" : "NULL")); + + break; + + case class_type: + { + classDef *cd = ad->u.cd; + int needNew = needNewInstance(ad); + + if (needNew) + prcode(fp, +" %s sipConvertFromNewInstance(", prefix); + else + prcode(fp, +" %s sipConvertFromInstance(", prefix); + + if (isConstArg(ad)) + prcode(fp,"const_cast<%b *>(%s)",ad,vname); + else + prcode(fp,"%s",vname); + + prcode(fp,",sipClass_%C,",classFQCName(cd)); + + if (needNew) + prcode(fp,"NULL"); + else + prcode(fp,"%s\n" + , (isTransferredBack(ad) ? "Py_None" : "NULL")); + + prcode(fp,");\n" + ); + } + + break; + + case bool_type: + case cbool_type: + prcode(fp, +" %s PyBool_FromLong(%s);\n" + ,prefix,vname); + + break; + + case sstring_type: + case ustring_type: + case string_type: + if (ad->nrderefs == 0) + prcode(fp, +" %s PyString_FromStringAndSize(%s&%s,1);\n" + ,prefix,(ad->atype != string_type) ? "(char *)" : "",vname); + else + prcode(fp, +" if (%s == NULL)\n" +" {\n" +" Py_INCREF(Py_None);\n" +" return Py_None;\n" +" }\n" +"\n" +" %s PyString_FromString(%s%s);\n" + ,vname + ,prefix,(ad->atype != string_type) ? "(char *)" : "",vname); + + break; + + case wstring_type: + if (ad->nrderefs == 0) + prcode(fp, +" %s PyUnicode_FromWideChar(&%s,1);\n" + , prefix, vname); + else + prcode(fp, +" if (%s == NULL)\n" +" {\n" +" Py_INCREF(Py_None);\n" +" return Py_None;\n" +" }\n" +"\n" +" %s PyUnicode_FromWideChar(%s,(SIP_SSIZE_T)wcslen(%s));\n" + , vname + , prefix, vname, vname); + + break; + + case enum_type: + if (ad->u.ed->fqcname != NULL) + { + prcode(fp, +" %s sipConvertFromNamedEnum(%s,sipEnum_%C);\n" + ,prefix,vname,ad->u.ed->fqcname); + + break; + } + + /* Drop through. */ + + case short_type: + case int_type: + case cint_type: + prcode(fp, +" %s PyInt_FromLong(%s);\n" + ,prefix,vname); + + break; + + case long_type: + prcode(fp, +" %s PyLong_FromLong(%s);\n" + ,prefix,vname); + + break; + + case ushort_type: + case uint_type: + case ulong_type: + prcode(fp, +" %s PyLong_FromUnsignedLong(%s);\n" + ,prefix,vname); + + break; + + case longlong_type: + prcode(fp, +" %s PyLong_FromLongLong(%s);\n" + ,prefix,vname); + + break; + + case ulonglong_type: + prcode(fp, +" %s PyLong_FromUnsignedLongLong(%s);\n" + ,prefix,vname); + + break; + + case void_type: + prcode(fp, +" %s sipConvertFromVoidPtr(", prefix); + + if (isConstArg(ad)) + prcode(fp, "const_cast<void *>(%s)", vname); + else + prcode(fp, "%s", vname); + + prcode(fp, ");\n" + ); + + break; + + case struct_type: + prcode(fp, +" %s sipConvertFromVoidPtr(",prefix); + + if (isConstArg(ad)) + prcode(fp,"const_cast<%b *>(%s)",ad,vname); + else + prcode(fp,"%s",vname); + + prcode(fp,");\n" + ); + + break; + + case float_type: + case cfloat_type: + prcode(fp, +" %s PyFloat_FromDouble((double)%s);\n" + ,prefix,vname); + + break; + + case double_type: + case cdouble_type: + prcode(fp, +" %s PyFloat_FromDouble(%s);\n" + ,prefix,vname); + + break; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + prcode(fp, +" %s %s;\n" + ,prefix,vname); + + break; + } +} + + +/* + * Return the format character used by sipBuildResult() for a particular type. + */ +static char getBuildResultFormat(argDef *ad) +{ + switch (ad->atype) + { + case mapped_type: + return 'D'; + + case class_type: + if (needNewInstance(ad)) + return 'B'; + + return 'C'; + + case bool_type: + case cbool_type: + return 'b'; + + case sstring_type: + case ustring_type: + case string_type: + return (ad->nrderefs > (isOutArg(ad) ? 1 : 0)) ? 's' : 'c'; + + case wstring_type: + return (ad->nrderefs > (isOutArg(ad) ? 1 : 0)) ? 'x' : 'w'; + + case enum_type: + return (ad->u.ed->fqcname != NULL) ? 'E' : 'e'; + + case short_type: + return 'h'; + + case ushort_type: + return 't'; + + case int_type: + case cint_type: + return 'i'; + + case uint_type: + return 'u'; + + case long_type: + return 'l'; + + case ulong_type: + return 'm'; + + case longlong_type: + return 'n'; + + case ulonglong_type: + return 'o'; + + case void_type: + case struct_type: + return 'V'; + + case float_type: + case cfloat_type: + return 'f'; + + case double_type: + case cdouble_type: + return 'd'; + + case pyobject_type: + case pytuple_type: + case pylist_type: + case pydict_type: + case pycallable_type: + case pyslice_type: + case pytype_type: + return 'R'; + } + + /* We should never get here. */ + return ' '; +} + + +/* + * Generate a function call. + */ +static void generateFunctionCall(classDef *cd,classDef *ocd,overDef *od, + int deref, FILE *fp) +{ + int needsNew, error_flag = FALSE, newline, is_result, a, deltemps; + argDef *res = &od->pysig.result, orig_res; + + prcode(fp, +" {\n" + ); + + /* + * If there is no shadow class then protected methods can never be + * called. + */ + if (isProtected(od) && !hasShadow(cd)) + { + prcode(fp, +" /* Never reached. */\n" +" }\n" + ); + + return; + } + + /* Save the full result type as we may want to fiddle with it. */ + orig_res = *res; + + /* See if we need to make a copy of the result on the heap. */ + if ((res->atype == class_type || res->atype == mapped_type) && + !isReference(res) && + res->nrderefs == 0) + { + needsNew = TRUE; + resetIsConstArg(res); + } + else + needsNew = FALSE; + + /* See if sipRes is needed. */ + is_result = (!isInplaceNumberSlot(od->common) && + !isInplaceSequenceSlot(od->common) && + (res->atype != void_type || res->nrderefs != 0)); + + newline = FALSE; + + if (is_result) + { + prcode(fp, +" "); + + generateNamedValueType(res,"sipRes",fp); + + /* + * The typical %MethodCode usually causes a compiler warning, + * so we initialise the result in that case to try and suppress + * it. + */ + if (od->methodcode != NULL) + { + prcode(fp," = "); + + generateCastZero(res,fp); + } + + prcode(fp,";\n" + ); + + newline = TRUE; + } + + deltemps = TRUE; + + for (a = 0; a < od->pysig.nrArgs; ++a) + { + argDef *ad = &od->pysig.args[a]; + + /* + * If we have an In,Out argument that has conversion code then we delay + * the destruction of any temporary variables until after we have + * converted the outputs. + */ + if (isInArg(ad) && isOutArg(ad) && hasConvertToCode(ad) && deltemps) + { + deltemps = FALSE; + + prcode(fp, +" PyObject *sipResult;\n" + ); + + newline = TRUE; + } + + /* + * If we are returning a class via an output only reference or pointer + * then we need an instance on the heap. + */ + if (needNewInstance(ad)) + { + prcode(fp, +" a%d = new %b();\n" + ,a,ad); + + newline = TRUE; + } + } + + if (od->methodcode != NULL) + { + /* See if the handwritten code seems to be using the error flag. */ + if (needErrorFlag(od->methodcode)) + { + prcode(fp, +" int sipIsErr = 0;\n" + ); + + newline = TRUE; + error_flag = TRUE; + } + } + + if (newline) + prcode(fp, +"\n" + ); + + /* If it is abstract make sure that self was bound. */ + if (isAbstract(od)) + prcode(fp, +" if (sipSelfWasArg)\n" +" {\n" +" sipAbstractMethod(%N,%N);\n" +" return NULL;\n" +" }\n" +"\n" + , cd->iff->name, od->common->pyname); + + /* Call any pre-hook. */ + if (od->prehook != NULL) + prcode(fp, +" sipCallHook(\"%s\");\n" +"\n" + ,od->prehook); + + if (od->methodcode != NULL) + generateCppCodeBlock(od->methodcode,fp); + else + { + int rgil = ((release_gil || isReleaseGIL(od)) && !isHoldGIL(od)); + + if (needsNew && generating_c) + { + prcode(fp, +" if ((sipRes = (%b *)sipMalloc(sizeof (%b))) == NULL)\n" +" {\n" + ,res,res); + + gc_ellipsis(&od->pysig, fp); + + prcode(fp, +" return NULL;\n" +" }\n" +"\n" + ); + } + + if (rgil) + prcode(fp, +" Py_BEGIN_ALLOW_THREADS\n" + ); + + generateTry(od->exceptions,fp); + + prcode(fp, +" "); + + if (od->common->slot != cmp_slot && is_result) + { + /* Construct a copy on the heap if needed. */ + if (needsNew) + { + if (generating_c) + prcode(fp,"*sipRes = "); + else + prcode(fp,"sipRes = new %b(",res); + } + else + prcode(fp,"sipRes = "); + + /* See if we need the address of the result. */ + if ((res->atype == class_type || res->atype == mapped_type) && isReference(res)) + prcode(fp,"&"); + } + + switch (od->common->slot) + { + case no_slot: + generateCppFunctionCall(cd,ocd,od,fp); + break; + + case getitem_slot: + prcode(fp, "(*sipCpp)["); + generateSlotArg(&od->pysig, 0, fp); + prcode(fp,"]"); + break; + + case call_slot: + prcode(fp, "(*sipCpp)("); + generateArgs(od->cppsig,Call,fp); + prcode(fp,")"); + break; + + case int_slot: + case long_slot: + case float_slot: + prcode(fp, "*sipCpp"); + break; + + case add_slot: + generateNumberSlotCall(od,"+",fp); + break; + + case concat_slot: + generateBinarySlotCall(od,"+",deref,fp); + break; + + case sub_slot: + generateNumberSlotCall(od,"-",fp); + break; + + case mul_slot: + generateNumberSlotCall(od,"*",fp); + break; + + case repeat_slot: + generateBinarySlotCall(od,"*",deref,fp); + break; + + case div_slot: + generateNumberSlotCall(od,"/",fp); + break; + + case mod_slot: + generateNumberSlotCall(od,"%",fp); + break; + + case and_slot: + generateNumberSlotCall(od,"&",fp); + break; + + case or_slot: + generateNumberSlotCall(od,"|",fp); + break; + + case xor_slot: + generateNumberSlotCall(od,"^",fp); + break; + + case lshift_slot: + generateNumberSlotCall(od,"<<",fp); + break; + + case rshift_slot: + generateNumberSlotCall(od,">>",fp); + break; + + case iadd_slot: + case iconcat_slot: + generateBinarySlotCall(od,"+=",deref,fp); + break; + + case isub_slot: + generateBinarySlotCall(od,"-=",deref,fp); + break; + + case imul_slot: + case irepeat_slot: + generateBinarySlotCall(od,"*=",deref,fp); + break; + + case idiv_slot: + generateBinarySlotCall(od,"/=",deref,fp); + break; + + case imod_slot: + generateBinarySlotCall(od,"%=",deref,fp); + break; + + case iand_slot: + generateBinarySlotCall(od,"&=",deref,fp); + break; + + case ior_slot: + generateBinarySlotCall(od,"|=",deref,fp); + break; + + case ixor_slot: + generateBinarySlotCall(od,"^=",deref,fp); + break; + + case ilshift_slot: + generateBinarySlotCall(od,"<<=",deref,fp); + break; + + case irshift_slot: + generateBinarySlotCall(od,">>=",deref,fp); + break; + + case invert_slot: + prcode(fp, "~(*sipCpp)"); + break; + + case lt_slot: + generateBinarySlotCall(od,"<",deref,fp); + break; + + case le_slot: + generateBinarySlotCall(od,"<=",deref,fp); + break; + + case eq_slot: + generateBinarySlotCall(od,"==",deref,fp); + break; + + case ne_slot: + generateBinarySlotCall(od,"!=",deref,fp); + break; + + case gt_slot: + generateBinarySlotCall(od,">",deref,fp); + break; + + case ge_slot: + generateBinarySlotCall(od,">=",deref,fp); + break; + + case neg_slot: + prcode(fp, "-(*sipCpp)"); + break; + + case pos_slot: + prcode(fp, "+(*sipCpp)"); + break; + + case cmp_slot: + prcode(fp,"if "); + generateBinarySlotCall(od,"<",deref,fp); + prcode(fp,"\n" +" sipRes = -1;\n" +" else if "); + generateBinarySlotCall(od,">",deref,fp); + prcode(fp,"\n" +" sipRes = 1;\n" +" else\n" +" sipRes = 0"); + + break; + } + + if (needsNew && !generating_c) + prcode(fp,")"); + + prcode(fp,";\n" + ); + + generateCatch(od->exceptions, &od->pysig, fp); + + if (rgil) + prcode(fp, +" Py_END_ALLOW_THREADS\n" + ); + } + + /* Handle /TransferThis/ for non-factory methods. */ + if (!isFactory(od)) + for (a = 0; a < od->pysig.nrArgs; ++a) + { + argDef *ad = &od->pysig.args[a]; + + if (!isInArg(ad)) + continue; + + if (isThisTransferred(ad)) + { + prcode(fp, +"\n" +" if (sipOwner)\n" +" sipTransferTo(sipSelf, (PyObject *)sipOwner);\n" +" else\n" +" sipTransferBack(sipSelf);\n" + ); + + break; + } + } + + gc_ellipsis(&od->pysig, fp); + + if (deltemps) + deleteTemps(&od->pysig, fp); + + prcode(fp, +"\n" + ); + + /* Handle sipIsErr if it was used. */ + if (error_flag) + prcode(fp, +" if (sipIsErr)\n" +" return %s;\n" +"\n" + ,((isVoidReturnSlot(od->common) || isIntReturnSlot(od->common) || isLongReturnSlot(od->common)) ? "-1" : "0")); + + /* Call any post-hook. */ + if (od->posthook != NULL) + prcode(fp, +"\n" +" sipCallHook(\"%s\");\n" + ,od->posthook); + + if (isVoidReturnSlot(od->common)) + prcode(fp, +" return 0;\n" + ); + else if (isInplaceNumberSlot(od->common) || isInplaceSequenceSlot(od->common)) + prcode(fp, +" Py_INCREF(sipSelf);\n" +" return sipSelf;\n" + ); + else if (isIntReturnSlot(od->common) || isLongReturnSlot(od->common)) + prcode(fp, +" return sipRes;\n" + ); + else + { + generateHandleResult(od,needsNew,(deltemps ? "return" : "sipResult ="),fp); + + /* Delete the temporaries now if we haven't already done so. */ + if (!deltemps) + { + deleteTemps(&od->pysig, fp); + + prcode(fp, +"\n" +" return sipResult;\n" + ); + } + } + + prcode(fp, +" }\n" + ); + + /* Restore the full type of the result. */ + *res = orig_res; +} + + +/* + * Generate a call to a C++ function. + */ +static void generateCppFunctionCall(classDef *cd,classDef *ocd,overDef *od, + FILE *fp) +{ + char *mname = od->cppname; + int parens = 1; + + /* + * If the function is protected then call the public wrapper. If it is + * virtual then call explicit scoped function if "self" was passed as + * the first argument. + */ + + if (cd == NULL) + prcode(fp,"%s(",mname); + else if (cd->iff->type == namespace_iface) + prcode(fp,"%S::%s(",classFQCName(cd),mname); + else if (isStatic(od)) + { + if (isProtected(od)) + prcode(fp,"sip%C::sipProtect_%s(",classFQCName(cd),mname); + else + prcode(fp,"%S::%s(",classFQCName(ocd),mname); + } + else if (isProtected(od)) + { + if (!isAbstract(od) && (isVirtual(od) || isVirtualReimp(od))) + { + prcode(fp, "sipCpp->sipProtectVirt_%s(sipSelfWasArg", mname); + + if (od->cppsig->nrArgs > 0) + prcode(fp, ","); + } + else + prcode(fp, "sipCpp->sipProtect_%s(", mname); + } + else if (!isAbstract(od) && (isVirtual(od) || isVirtualReimp(od))) + { + prcode(fp, "(sipSelfWasArg ? sipCpp->%U::%s(", ocd, mname); + generateArgs(od->cppsig, Call, fp); + prcode(fp, ") : sipCpp->%s(", mname); + ++parens; + } + else + prcode(fp, "sipCpp->%s(", mname); + + generateArgs(od->cppsig, Call, fp); + + while (parens--) + prcode(fp, ")"); +} + + +/* + * Generate argument to a slot. + */ +static void generateSlotArg(signatureDef *sd, int argnr, FILE *fp) +{ + argDef *ad; + int deref; + + ad = &sd->args[argnr]; + deref = ((ad->atype == class_type || ad->atype == mapped_type) && ad->nrderefs == 0); + + prcode(fp, "%sa%d", (deref ? "*" : ""), argnr); +} + + +/* + * Generate the call to a binary (non-number) slot method. + */ +static void generateBinarySlotCall(overDef *od, char *op, int deref, FILE *fp) +{ + if (deref) + prcode(fp, "((*sipCpp) %s ", op); + else + prcode(fp, "(sipCpp %s ", op); + + generateSlotArg(&od->pysig, 0, fp); + prcode(fp, ")"); +} + + +/* + * Generate the call to a binary number slot method. + */ +static void generateNumberSlotCall(overDef *od, char *op, FILE *fp) +{ + prcode(fp, "("); + generateSlotArg(&od->pysig, 0, fp); + prcode(fp, " %s ", op); + generateSlotArg(&od->pysig, 1, fp); + prcode(fp, ")"); +} + + +/* + * Generate the argument variables for a member function/constructor/operator. + */ +static int generateArgParser(sipSpec *pt, signatureDef *sd, classDef *cd, + ctorDef *ct, overDef *od, int secCall, FILE *fp) +{ + int a, isQtSlot, optargs, arraylenarg, sigarg, handle_self; + int slotconarg, slotdisarg, need_owner; + + /* If the class is just a namespace, then ignore it. */ + + if (cd != NULL && cd->iff->type == namespace_iface) + cd = NULL; + + handle_self = (od != NULL && od->common->slot == no_slot && !isStatic(od) && cd != NULL); + + /* Assume there isn't a Qt slot. */ + + isQtSlot = FALSE; + + /* + * Generate the local variables that will hold the parsed arguments and + * values returned via arguments. + */ + + sigarg = -1; + need_owner = FALSE; + + for (a = 0; a < sd->nrArgs; ++a) + { + argDef *ad = &sd->args[a]; + + switch (ad->atype) + { + case signal_type: + sigarg = a; + break; + + case rxcon_type: + case rxdis_type: + isQtSlot = TRUE; + break; + + case slotcon_type: + slotconarg = a; + break; + + case slotdis_type: + slotdisarg = a; + break; + } + + if (isArraySize(ad)) + arraylenarg = a; + + generateVariable(ad,a,fp); + + if (isThisTransferred(ad)) + need_owner = TRUE; + } + + if (od != NULL && need_owner) + prcode(fp, +" sipWrapper *sipOwner = 0;\n" + ); + + if (handle_self) + { + if (isProtected(od) && hasShadow(cd)) + prcode(fp, +" sip%C *sipCpp;\n" + ,classFQCName(cd)); + else + prcode(fp, +" %U *sipCpp;\n" + ,cd); + + prcode(fp, +"\n" + ); + } + else if (sd->nrArgs != 0) + prcode(fp, +"\n" + ); + + /* Generate the call to the parser function. */ + if (od != NULL && isNumberSlot(od->common)) + prcode(fp, +" if (sipParsePair(%ssipArgsParsed,sipArg0,sipArg1,\"", (ct != NULL ? "" : "&")); + else + prcode(fp, +" if (sipParseArgs(%ssipArgsParsed,sipArg%s,\"", (ct != NULL ? "" : "&"), (od == NULL || od->common->slot == no_slot || isMultiArgSlot(od->common)) ? "s" : ""); + + /* Generate the format string. */ + optargs = FALSE; + + if (handle_self) + prcode(fp,"%c",(isProtected(od) ? 'p' : 'B')); + else if (isQtSlot && od == NULL) + prcode(fp,"C"); + + for (a = 0; a < sd->nrArgs; ++a) + { + char *fmt = ""; + argDef *ad = &sd->args[a]; + + if (!isInArg(ad)) + continue; + + if (ad->defval != NULL && !optargs) + { + prcode(fp,"|"); + optargs = TRUE; + } + + switch (ad->atype) + { + case sstring_type: + case ustring_type: + case string_type: + if (ad->nrderefs == 0 || (isOutArg(ad) && ad->nrderefs == 1)) + fmt = "c"; + else if (isArray(ad)) + fmt = "a"; + else + fmt = "s"; + + break; + + case wstring_type: + if (ad->nrderefs == 0 || (isOutArg(ad) && ad->nrderefs == 1)) + fmt = "w"; + else if (isArray(ad)) + fmt = "A"; + else + fmt = "x"; + + break; + + case enum_type: + fmt = (ad->u.ed->fqcname != NULL) ? "E" : "e"; + break; + + case bool_type: + fmt = "b"; + break; + + case cbool_type: + fmt = "Xb"; + break; + + case int_type: + if (!isArraySize(ad)) + fmt = "i"; + + break; + + case uint_type: + if (!isArraySize(ad)) + fmt = "u"; + + break; + + case cint_type: + fmt = "Xi"; + break; + + case short_type: + if (!isArraySize(ad)) + fmt = "h"; + + break; + + case ushort_type: + if (!isArraySize(ad)) + fmt = "t"; + + break; + + case long_type: + if (!isArraySize(ad)) + fmt = "l"; + + break; + + case ulong_type: + if (!isArraySize(ad)) + fmt = "m"; + + break; + + case longlong_type: + if (!isArraySize(ad)) + fmt = "n"; + + break; + + case ulonglong_type: + if (!isArraySize(ad)) + fmt = "o"; + + break; + + case struct_type: + case void_type: + fmt = "v"; + break; + + case float_type: + fmt = "f"; + break; + + case cfloat_type: + fmt = "Xf"; + break; + + case double_type: + fmt = "d"; + break; + + case cdouble_type: + fmt = "Xd"; + break; + + case signal_type: + fmt = "G"; + break; + + case slot_type: + fmt = "S"; + break; + + case anyslot_type: + fmt = "U"; + break; + + case slotcon_type: + case slotdis_type: + fmt = (secCall ? "" : "S"); + break; + + case rxcon_type: + fmt = (secCall ? "y" : "q"); + break; + + case rxdis_type: + fmt = (secCall ? "Y" : "Q"); + break; + + case mapped_type: + fmt = getSubFormatChar('M',ad); + break; + + case class_type: + fmt = getSubFormatChar('J', ad); + break; + + case pyobject_type: + fmt = getSubFormatChar('P',ad); + break; + + case pytuple_type: + case pylist_type: + case pydict_type: + case pyslice_type: + case pytype_type: + fmt = (isAllowNone(ad) ? "N" : "T"); + break; + + case pycallable_type: + fmt = (isAllowNone(ad) ? "H" : "F"); + break; + + case qobject_type: + fmt = "R"; + break; + + case ellipsis_type: + fmt = "W"; + break; + } + + prcode(fp,fmt); + } + + prcode(fp,"\""); + + /* Generate the parameters corresponding to the format string. */ + + if (handle_self) + prcode(fp,",&sipSelf,sipClass_%C,&sipCpp",classFQCName(cd)); + else if (isQtSlot && od == NULL) + prcode(fp,",sipSelf"); + + for (a = 0; a < sd->nrArgs; ++a) + { + argDef *ad = &sd->args[a]; + + if (!isInArg(ad)) + continue; + + switch (ad->atype) + { + case mapped_type: + prcode(fp,",sipMappedType_%T,&a%d,&a%dState",ad,a,a); + break; + + case class_type: + prcode(fp, ",sipClass_%T,&a%d", ad, a); + + if (isThisTransferred(ad)) + prcode(fp, ",%ssipOwner", (ct != NULL ? "" : "&")); + else if (isGetWrapper(ad)) + prcode(fp, ",&a%dWrapper", a); + + if (ad->u.cd->convtocode != NULL && !isConstrained(ad)) + prcode(fp, ",&a%dState", a); + + break; + + case rxcon_type: + { + if (sigarg > 0) + prcode(fp,",a%d",sigarg); + else + { + prcode(fp,",\"("); + + generateArgs(sd->args[slotconarg].u.sa,Declaration,fp); + + prcode(fp,")\""); + } + + prcode(fp,",&a%d,&a%d",a,slotconarg); + + break; + } + + case rxdis_type: + { + prcode(fp,",\"("); + + generateArgs(sd->args[slotdisarg].u.sa,Declaration,fp); + + prcode(fp,")\",&a%d,&a%d",a,slotdisarg); + + break; + } + + case slotcon_type: + case slotdis_type: + if (!secCall) + prcode(fp,",&a%d",a); + + break; + + case anyslot_type: + prcode(fp, ",&a%dName,&a%dCallable", a, a); + break; + + case pytuple_type: + prcode(fp,",&PyTuple_Type,&a%d",a); + break; + + case pylist_type: + prcode(fp,",&PyList_Type,&a%d",a); + break; + + case pydict_type: + prcode(fp,",&PyDict_Type,&a%d",a); + break; + + case pyslice_type: + prcode(fp,",&PySlice_Type,&a%d",a); + break; + + case pytype_type: + prcode(fp,",&PyType_Type,&a%d",a); + break; + + case enum_type: + if (ad->u.ed->fqcname != NULL) + prcode(fp,",sipEnum_%C",ad->u.ed->fqcname); + + prcode(fp,",&a%d",a); + break; + + default: + if (!isArraySize(ad)) + prcode(fp,",&a%d",a); + + if (isArray(ad)) + prcode(fp,",&a%d",arraylenarg); + } + } + + prcode(fp,"))\n"); + + return isQtSlot; +} + + +/* + * Get the format character string for something that has sub-formats. + */ + +static char *getSubFormatChar(char fc,argDef *ad) +{ + static char fmt[3]; + char flags; + + fmt[0] = fc; + + flags = 0; + + if (fc != 'P' && ad->nrderefs == 0) + flags |= 0x01; + + if (isTransferred(ad)) + flags |= 0x02; + + if (isTransferredBack(ad)) + flags |= 0x04; + + if (fc == 'J') + { + if (isThisTransferred(ad)) + flags |= 0x20; + + if (isGetWrapper(ad)) + flags |= 0x08; + + if (ad->u.cd->convtocode == NULL || isConstrained(ad)) + flags |= 0x10; + } + + fmt[1] = '0' + flags; + + fmt[2] = '\0'; + + return fmt; +} + + +/* + * Return TRUE if a type has %ConvertToTypeCode. + */ +static int hasConvertToCode(argDef *ad) +{ + codeBlock *convtocode; + + if (ad->atype == class_type && !isConstrained(ad)) + convtocode = ad->u.cd->convtocode; + else if (ad->atype == mapped_type) + convtocode = ad->u.mtd->convtocode; + else + convtocode = NULL; + + return (convtocode != NULL); +} + + +/* + * Garbage collect any ellipsis argument. + */ +static void gc_ellipsis(signatureDef *sd, FILE *fp) +{ + if (sd->nrArgs > 0 && sd->args[sd->nrArgs - 1].atype == ellipsis_type) + prcode(fp, +"\n" +" Py_DECREF(a%d);\n" + , sd->nrArgs - 1); +} + + +/* + * Delete any temporary variables on the heap created by type convertors. + */ +static void deleteTemps(signatureDef *sd, FILE *fp) +{ + int a, first = TRUE; + + for (a = 0; a < sd->nrArgs; ++a) + { + argDef *ad = &sd->args[a]; + + if (!isInArg(ad)) + continue; + + if (ad->atype == wstring_type && ad->nrderefs == 1) + { + if (generating_c || !isConstArg(ad)) + prcode(fp, +" sipFree(a%d);\n" + , a); + else + prcode(fp, +" sipFree(const_cast<wchar_t *>(a%d));\n" + , a); + } + else if (hasConvertToCode(ad)) + { + const char *fstr, *sstr; + + if (ad->atype == mapped_type) + fstr = sstr = "MappedType"; + else + { + fstr = "Instance"; + sstr = "Class"; + } + + if (first) + { + prcode(fp, +"\n" + ); + + first = FALSE; + } + + if (generating_c || !isConstArg(ad)) + prcode(fp, +" sipRelease%s(a%d,sip%s_%T,a%dState);\n" + , fstr, a, sstr, ad, a); + else + prcode(fp, +" sipRelease%s(const_cast<%b *>(a%d),sip%s_%T,a%dState);\n" + , fstr, ad, a, sstr, ad, a); + } + } +} + + +/* + * Generate a C++ code block. + */ +static void generateCppCodeBlock(codeBlock *code,FILE *fp) +{ + int reset_line = FALSE; + codeBlock *cb; + + for (cb = code; cb != NULL; cb = cb->next) + { + char *cp; + + /* + * Fragmented fragments (possibly created when applying + * template types) don't have a filename. + */ + if ((cp = cb->filename) != NULL) + { + reset_line = TRUE; + + prcode(fp, +"#line %d \"", cb->linenr); + + while (*cp != '\0') + { + prcode(fp, "%c", *cp); + + if (*cp == '\\') + prcode(fp, "\\"); + + ++cp; + } + + prcode(fp, "\"\n" + ); + } + + prcode(fp, "%s", cb->frag); + } + + if (reset_line) + { + char *bn; + + /* Just use the base name. */ + + if ((bn = strrchr(currentFileName,'/')) != NULL) + ++bn; + else + bn = currentFileName; + + prcode(fp, +"#line %d \"%s\"\n" + ,currentLineNr + 1,bn); + } +} + + +/* + * Create a source file. + */ +static FILE *createCompilationUnit(sipSpec *pt, char *fname, char *description) +{ + FILE *fp = createFile(pt, fname, description); + + if (fp != NULL) + generateCppCodeBlock(pt->unitcode, fp); + + return fp; +} + + +/* + * Create a file with an optional standard header. + */ +static FILE *createFile(sipSpec *pt,char *fname,char *description) +{ + FILE *fp; + + /* Create the file. */ + + if ((fp = fopen(fname,"w")) == NULL) + fatal("Unable to create file \"%s\"\n",fname); + + /* The "stack" doesn't have to be very deep. */ + previousLineNr = currentLineNr; + currentLineNr = 1; + previousFileName = currentFileName; + currentFileName = fname; + + if (description != NULL) + { + int needComment; + codeBlock *cb; + time_t now; + + /* Write the header. */ + now = time(NULL); + + prcode(fp, +"/*\n" +" * %s\n" +" *\n" +" * Generated by SIP %s on %s" + ,description + ,sipVersion,ctime(&now)); + + if (pt->copying != NULL) + prcode(fp, +" *\n" + ); + + needComment = TRUE; + + for (cb = pt->copying; cb != NULL; cb = cb->next) + { + char *cp; + + for (cp = cb->frag; *cp != '\0'; ++cp) + { + if (needComment) + { + needComment = FALSE; + prcode(fp," * "); + } + + prcode(fp,"%c",*cp); + + if (*cp == '\n') + needComment = TRUE; + } + } + + prcode(fp, +" */\n" + ); + } + + return fp; +} + + +/* + * Close a file and report any errors. + */ +static void closeFile(FILE *fp) +{ + if (ferror(fp)) + fatal("Error writing to \"%s\"\n",currentFileName); + + if (fclose(fp)) + fatal("Error closing \"%s\"\n",currentFileName); + + currentLineNr = previousLineNr; + currentFileName = previousFileName; +} + + +/* + * Print formatted code. + */ +void prcode(FILE *fp, const char *fmt, ...) +{ + char ch; + va_list ap; + + prcode_last = fmt; + + va_start(ap,fmt); + + while ((ch = *fmt++) != '\0') + if (ch == '%') + { + ch = *fmt++; + + switch (ch) + { + case 'c': + { + char c = (char)va_arg(ap,int); + + if (c == '\n') + ++currentLineNr; + + fputc(c,fp); + break; + } + + case 's': + { + const char *cp = va_arg(ap,const char *); + + while (*cp != '\0') + { + if (*cp == '\n') + ++currentLineNr; + + fputc(*cp,fp); + ++cp; + } + + break; + } + + case 'l': + fprintf(fp,"%ld",va_arg(ap,long)); + break; + + case 'u': + fprintf(fp,"%u",va_arg(ap,unsigned)); + break; + + case 'd': + fprintf(fp,"%d",va_arg(ap,int)); + break; + + case 'g': + fprintf(fp,"%g",va_arg(ap,double)); + break; + + case 'x': + fprintf(fp,"0x%08x",va_arg(ap,unsigned)); + break; + + case '\0': + fputc('%',fp); + --fmt; + break; + + case '\n': + fputc('\n',fp); + ++currentLineNr; + break; + + case 'b': + { + argDef *ad, orig; + + ad = va_arg(ap,argDef *); + orig = *ad; + + resetIsConstArg(ad); + resetIsReference(ad); + ad->nrderefs = 0; + + generateBaseType(ad,fp); + + *ad = orig; + + break; + } + + case 'M': + prcode_xml = !prcode_xml; + break; + + case 'B': + generateBaseType(va_arg(ap,argDef *),fp); + break; + + case 'T': + prTypeName(fp,va_arg(ap,argDef *),FALSE); + break; + + case 'I': + { + int indent = va_arg(ap,int); + + while (indent-- > 0) + fputc('\t',fp); + + break; + } + + case 'N': + { + nameDef *nd = va_arg(ap,nameDef *); + + fprintf(fp,"sipNm_%s_%s",nd->module->name,nd->text); + break; + } + + case 'E': + { + enumDef *ed = va_arg(ap,enumDef *); + + if (ed->fqcname == NULL) + fprintf(fp,"int"); + else if (isProtectedEnum(ed)) + { + fprintf(fp,"sip"); + prScopedName(fp,classFQCName(ed->pcd),"_"); + + fprintf(fp,"::sip%s",scopedNameTail(ed->fqcname)); + } + else + prScopedName(fp,ed->fqcname,"::"); + + break; + } + + case 'F': + prScopedName(fp,va_arg(ap,scopedNameDef *),""); + break; + + case 'C': + prScopedName(fp,va_arg(ap,scopedNameDef *),"_"); + break; + + case 'S': + if (generating_c) + fprintf(fp,"struct "); + + prScopedName(fp,va_arg(ap,scopedNameDef *),"::"); + break; + + case 'U': + if (generating_c) + fprintf(fp,"struct "); + + prScopedClassName(fp,va_arg(ap,classDef *),"::"); + break; + + case 'O': + prOverloadName(fp, va_arg(ap, overDef *)); + break; + + case 'P': + { + classDef *ecd = va_arg(ap, classDef *); + const char *pyname = va_arg(ap, const char *); + + prScopedPythonName(fp, ecd, pyname); + break; + } + + case 'X': + generateThrowSpecifier(va_arg(ap,throwArgs *),fp); + break; + + default: + fputc(ch,fp); + } + } + else if (ch == '\n') + { + fputc('\n',fp); + ++currentLineNr; + } + else + fputc(ch,fp); + + va_end(ap); +} + + +/* + * Generate the C++ name of an overloaded function. + */ +void prOverloadName(FILE *fp, overDef *od) +{ + char *pt1, *pt2; + + pt1 = "operator"; + + switch (od->common->slot) + { + case add_slot: + pt2 = "+"; + break; + + case sub_slot: + pt2 = "-"; + break; + + case mul_slot: + pt2 = "*"; + break; + + case div_slot: + pt2 = "/"; + break; + + case mod_slot: + pt2 = "%"; + break; + + case and_slot: + pt2 = "&"; + break; + + case or_slot: + pt2 = "|"; + break; + + case xor_slot: + pt2 = "^"; + break; + + case lshift_slot: + pt2 = "<<"; + break; + + case rshift_slot: + pt2 = ">>"; + break; + + case iadd_slot: + pt2 = "+="; + break; + + case isub_slot: + pt2 = "-="; + break; + + case imul_slot: + pt2 = "*="; + break; + + case idiv_slot: + pt2 = "/="; + break; + + case imod_slot: + pt2 = "%="; + break; + + case iand_slot: + pt2 = "&="; + break; + + case ior_slot: + pt2 = "|="; + break; + + case ixor_slot: + pt2 = "^="; + break; + + case ilshift_slot: + pt2 = "<<="; + break; + + case irshift_slot: + pt2 = ">>="; + break; + + case invert_slot: + pt2 = "~"; + break; + + case call_slot: + pt2 = "()"; + break; + + case getitem_slot: + pt2 = "[]"; + break; + + case lt_slot: + pt2 = "<"; + break; + + case le_slot: + pt2 = "<="; + break; + + case eq_slot: + pt2 = "=="; + break; + + case ne_slot: + pt2 = "!="; + break; + + case gt_slot: + pt2 = ">"; + break; + + case ge_slot: + pt2 = ">="; + break; + + default: + pt1 = ""; + pt2 = od->cppname; + } + + fprintf(fp, "%s%s", pt1, pt2); +} + + +/* + * Generate a scoped name with the given separator string. + */ +static void prScopedName(FILE *fp,scopedNameDef *snd,char *sep) +{ + while (snd != NULL) + { + fprintf(fp,"%s",snd->name); + + if ((snd = snd->next) != NULL) + fprintf(fp,"%s",sep); + } +} + + +/* + * Generate a scoped class name with the given separator string. At the moment + * this provides (probably) broken support for protected classes. + */ +static void prScopedClassName(FILE *fp,classDef *cd,char *sep) +{ + scopedNameDef *snd = classFQCName(cd); + + while (snd != NULL) + { + if (isProtectedClass(cd)) + fprintf(fp,"sip"); + + fprintf(fp,"%s",snd->name); + + if ((snd = snd->next) != NULL) + fprintf(fp,"%s",sep); + } +} + + +/* + * Generate a scoped Python name. + */ +void prScopedPythonName(FILE *fp, classDef *scope, const char *pyname) +{ + if (scope != NULL) + { + prScopedPythonName(fp, scope->ecd, NULL); + fprintf(fp, "%s.", scope->pyname); + } + + if (pyname != NULL) + fprintf(fp, "%s", pyname); +} + + +/* + * Generate a type name to be used as part of an identifier name. + */ +static void prTypeName(FILE *fp,argDef *ad,int intmpl) +{ + if (intmpl) + { + char buf[10]; + int flgs; + + /* We use numbers so they don't conflict with names. */ + + sprintf(buf,"%02d",ad->atype); + + flgs = 0; + + if (isConstArg(ad)) + flgs += 1; + + if (isReference(ad)) + flgs += 2; + + prcode(fp,"%s%d%d",buf,flgs,ad->nrderefs); + } + + /* Now add a name if there is one associated with the type. */ + + switch (ad->atype) + { + case struct_type: + prcode(fp,"%C",ad->u.sname); + break; + + case defined_type: + prcode(fp,"%C",ad->u.snd); + break; + + case enum_type: + prcode(fp,"%C",ad->u.ed->fqcname); + break; + + case mapped_type: + prTypeName(fp,&ad->u.mtd->type,intmpl); + break; + + case class_type: + prcode(fp,"%C",classFQCName(ad->u.cd)); + break; + + case template_type: + { + int a; + templateDef *td = ad->u.td; + + prcode(fp,"%C",td->fqname); + + for (a = 0; a < td->types.nrArgs; ++a) + { + prcode(fp,"_"); + prTypeName(fp,&td->types.args[a],TRUE); + } + + break; + } + } +} + + +/* + * Return TRUE if handwritten code uses the error flag. + */ +static int needErrorFlag(codeBlock *cb) +{ + return usedInCode(cb, "sipIsErr"); +} + + +/* + * Return TRUE if the argument type means an instance needs to be created on + * the heap to pass back to Python. + */ +static int needNewInstance(argDef *ad) +{ + return ((ad->atype == mapped_type || ad->atype == class_type) && + ((isReference(ad) && ad->nrderefs == 0) || (!isReference(ad) && ad->nrderefs == 1)) && + !isInArg(ad) && isOutArg(ad)); +} + + +/* + * Reset and save any argument flags so that the signature will be rendered + * exactly as defined in C++. + */ +static void normaliseArgs(signatureDef *sd) +{ + int a; + argDef *ad = sd->args; + + for (a = 0; a < sd->nrArgs; ++a) + { + if (ad->atype == class_type && isProtectedClass(ad->u.cd)) + { + resetIsProtectedClass(ad->u.cd); + setWasProtectedClass(ad->u.cd); + } + else if (ad->atype == enum_type && isProtectedEnum(ad->u.ed)) + { + resetIsProtectedEnum(ad->u.ed); + setWasProtectedEnum(ad->u.ed); + } + + ++ad; + } +} + + +/* + * Restore any argument flags modified by normaliseArgs(). + */ +static void restoreArgs(signatureDef *sd) +{ + int a; + argDef *ad = sd->args; + + for (a = 0; a < sd->nrArgs; ++a) + { + if (ad->atype == class_type && wasProtectedClass(ad->u.cd)) + { + resetWasProtectedClass(ad->u.cd); + setIsProtectedClass(ad->u.cd); + } + else if (ad->atype == enum_type && wasProtectedEnum(ad->u.ed)) + { + resetWasProtectedEnum(ad->u.ed); + setIsProtectedEnum(ad->u.ed); + } + + ++ad; + } +} + + +/* + * Return TRUE if a dealloc function is needed for a class. + */ +static int needDealloc(classDef *cd) +{ + if (cd->iff->type == namespace_iface) + return FALSE; + + /* All of these conditions cause some code to be generated. */ + + if (tracing) + return TRUE; + + if (generating_c) + return TRUE; + + if (cd->dealloccode != NULL) + return TRUE; + + if (isPublicDtor(cd)) + return TRUE; + + if (hasShadow(cd)) + return TRUE; + + return FALSE; +} + + +/* + * Return the argument name to use in a function definition for handwritten + * code. + */ +static const char *argName(const char *name, codeBlock *cb) +{ + static const char noname[] = ""; + + /* Always use the name in C code. */ + if (generating_c) + return name; + + /* Use the name if it is used in the handwritten code. */ + if (usedInCode(cb, name)) + return name; + + /* Don't use the name and avoid a compiler warning. */ + return noname; +} + + +/* + * Returns TRUE if a string is used in a code block. + */ +static int usedInCode(codeBlock *code, const char *str) +{ + while (code != NULL) + { + if (strstr(code->frag, str) != NULL) + return TRUE; + + code = code->next; + } + + return FALSE; +} + + +/* + * Generate an assignment statement from a void * variable to a class instance + * variable. + */ +static void generateClassFromVoid(classDef *cd, const char *cname, + const char *vname, FILE *fp) +{ + if (generating_c) + prcode(fp, "%S *%s = (%S *)%s", classFQCName(cd), cname, classFQCName(cd), vname); + else + prcode(fp, "%S *%s = reinterpret_cast<%S *>(%s)", classFQCName(cd), cname, classFQCName(cd), vname); +} + + +/* + * Generate an assignment statement from a void * variable to a mapped type + * variable. + */ +static void generateMappedTypeFromVoid(mappedTypeDef *mtd, const char *cname, + const char *vname, FILE *fp) +{ + if (generating_c) + prcode(fp, "%b *%s = (%b *)%s", &mtd->type, cname, &mtd->type, vname); + else + prcode(fp, "%b *%s = reinterpret_cast<%b *>(%s)", &mtd->type, cname, &mtd->type, vname); +} |