git-svn-id: https://swig.svn.sourceforge.net/svnroot/swig/trunk/SWIG@6490 626c5289-ae23-0410-ae9c-e8d60b6d4f22
237 lines
6.6 KiB
C
237 lines
6.6 KiB
C
/* -----------------------------------------------------------------------------
|
|
* util.c
|
|
*
|
|
* Parsing utilities
|
|
*
|
|
* Author(s) : David Beazley (beazley@cs.uchicago.edu)
|
|
*
|
|
* Copyright (C) 1999-2000. The University of Chicago
|
|
* See the file LICENSE for information on usage and redistribution.
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
char cvsroot_util_c[] = "$Header$";
|
|
|
|
#include "swig.h"
|
|
#include "cparse.h"
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* Swig_cparse_replace_descriptor()
|
|
*
|
|
* Replaces type descriptor string $descriptor() with the SWIG type descriptor
|
|
* string.
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
void Swig_cparse_replace_descriptor(String *s) {
|
|
char tmp[512];
|
|
String *arg = 0;
|
|
SwigType *t;
|
|
|
|
while (Strstr(s,"$descriptor(")) {
|
|
char *d = tmp;
|
|
int level = 0;
|
|
char *c = Strstr(s,"$descriptor(");
|
|
while (*c) {
|
|
if (*c == '(') level++;
|
|
if (*c == ')') {
|
|
level--;
|
|
if (level == 0) {
|
|
break;
|
|
}
|
|
}
|
|
*d = *c;
|
|
d++;
|
|
c++;
|
|
}
|
|
*d = 0;
|
|
arg = NewString(tmp+12);
|
|
t = Swig_cparse_type(arg);
|
|
Delete(arg);
|
|
arg = 0;
|
|
|
|
if (t) {
|
|
String *mangle;
|
|
String *descriptor;
|
|
|
|
mangle = SwigType_manglestr(t);
|
|
descriptor = NewStringf("SWIGTYPE%s",mangle);
|
|
SwigType_remember(t);
|
|
*d = ')';
|
|
d++;
|
|
*d = 0;
|
|
Replace(s,tmp,descriptor,DOH_REPLACE_ANY);
|
|
Delete(mangle);
|
|
Delete(descriptor);
|
|
} else {
|
|
Swig_error(Getfile(s),Getline(s),"Bad $descriptor() macro.\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* cparse_normalize_void()
|
|
*
|
|
* This function is used to replace arguments of the form (void) with empty
|
|
* arguments in C++
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
void cparse_normalize_void(Node *n) {
|
|
String *decl = Getattr(n,"decl");
|
|
Parm *parms = Getattr(n,"parms");
|
|
|
|
if (SwigType_isfunction(decl)) {
|
|
if ((ParmList_len(parms) == 1) && (SwigType_type(Getattr(parms,"type")) == T_VOID)) {
|
|
Replaceall(decl,"f(void).","f().");
|
|
Delattr(n,"parms");
|
|
}
|
|
}
|
|
}
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* int need_protected(Node* n, int dirprot_mode)
|
|
*
|
|
* Detects when we need to fully register the protected member.
|
|
*
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
int need_protected(Node* n, int dirprot_mode)
|
|
{
|
|
if (!(Swig_need_protected() || dirprot_mode)) return 0;
|
|
|
|
/* First, 'n' looks like a function */
|
|
if ((Strcmp(nodeType(n),"cdecl") == 0) &&
|
|
SwigType_isfunction(Getattr(n,"decl"))) {
|
|
String *storage = Getattr(n,"storage");
|
|
/* and the function is declared like virtual, or it has no
|
|
storage. This eliminates typedef, static and so on. */
|
|
return (!storage || (Strcmp(storage,"virtual") == 0));
|
|
} else if ((Strcmp(nodeType(n),"constructor") == 0)) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
* int need_name_warning(Node *n)
|
|
*
|
|
* Detects if a node needs name warnings
|
|
*
|
|
* ----------------------------------------------------------------------------- */
|
|
|
|
int need_name_warning(Node *n)
|
|
{
|
|
int need = 1;
|
|
/*
|
|
we don't use name warnings for:
|
|
- class forwards, no symbol is generated at the target language.
|
|
- template declarations, only for real instances using %template(name).
|
|
- typedefs, they have no effect at the target language.
|
|
*/
|
|
if (Strcmp(nodeType(n),"classforward") == 0) {
|
|
need = 0;
|
|
} else if (Getattr(n,"templatetype")) {
|
|
need = 0;
|
|
} else {
|
|
String *storage = Getattr(n,"storage");
|
|
if (storage && (Strcmp(storage,"typedef") == 0)) {
|
|
need = 0;
|
|
}
|
|
}
|
|
return need;
|
|
}
|
|
|
|
|
|
int are_equivalent_nodes(Node* a, Node* b, int a_inclass)
|
|
{
|
|
/* they must have the same type */
|
|
SwigType *ta = nodeType(a);
|
|
SwigType *tb = nodeType(b);
|
|
if (Cmp(ta, tb) != 0) return 0;
|
|
|
|
/* cdecl case */
|
|
if (Cmp(ta, "cdecl") == 0) {
|
|
/* typedef */
|
|
String *a_storage = Getattr(a,"storage");
|
|
String *b_storage = Getattr(b,"storage");
|
|
|
|
if ((Cmp(a_storage,"typedef") == 0)
|
|
|| (Cmp(b_storage,"typedef") == 0)) {
|
|
if (Cmp(a_storage, b_storage) == 0) {
|
|
String *a_type = (Getattr(a,"type"));
|
|
String *b_type = (Getattr(b,"type"));
|
|
if (Cmp(a_type, b_type) == 0) return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* static functions */
|
|
if ((Cmp(a_storage, "static") == 0)
|
|
|| (Cmp(b_storage, "static") == 0)) {
|
|
if (Cmp(a_storage, b_storage) != 0) return 0;
|
|
}
|
|
|
|
/* friend methods */
|
|
|
|
if (!a_inclass || (Cmp(a_storage,"friend") == 0)) {
|
|
/* check declaration */
|
|
|
|
String *a_decl = (Getattr(a,"decl"));
|
|
String *b_decl = (Getattr(b,"decl"));
|
|
if (Cmp(a_decl, b_decl) == 0) {
|
|
/* check return type */
|
|
String *a_type = (Getattr(a,"type"));
|
|
String *b_type = (Getattr(b,"type"));
|
|
if (Cmp(a_type, b_type) == 0) {
|
|
/* check parameters */
|
|
Parm *ap = (Getattr(a,"parms"));
|
|
Parm *bp = (Getattr(b,"parms"));
|
|
while (ap && bp) {
|
|
SwigType *at = Getattr(ap,"type");
|
|
SwigType *bt = Getattr(bp,"type");
|
|
if (Cmp(at, bt) != 0) return 0;
|
|
ap = nextSibling(ap);
|
|
bp = nextSibling(bp);
|
|
}
|
|
if (ap || bp) {
|
|
return 0;
|
|
} else {
|
|
Node *a_template = Getattr(a,"template");
|
|
Node *b_template = Getattr(b,"template");
|
|
/* Not equivalent if one is a template instantiation (via %template) and the other is a non-templated function */
|
|
if ((a_template && !b_template) || (!a_template && b_template)) return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/* %constant case */
|
|
String *a_storage = Getattr(a,"storage");
|
|
String *b_storage = Getattr(b,"storage");
|
|
if ((Cmp(a_storage, "%constant") == 0)
|
|
|| (Cmp(b_storage, "%constant") == 0)) {
|
|
if (Cmp(a_storage, b_storage) == 0) {
|
|
String *a_type = (Getattr(a,"type"));
|
|
String *b_type = (Getattr(b,"type"));
|
|
if ((Cmp(a_type, b_type) == 0)
|
|
&& (Cmp(Getattr(a,"value"), Getattr(b,"value")) == 0))
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int need_redefined_warn(Node* a, Node* b, int InClass)
|
|
{
|
|
String *a_symname = Getattr(a,"sym:name");
|
|
String *b_symname = Getattr(b,"sym:name");
|
|
/* always send a warning if a 'rename' is involved */
|
|
if ((a_symname && Cmp(a_symname,Getattr(a,"name")))
|
|
|| (b_symname && Cmp(b_symname,Getattr(b,"name"))))
|
|
return 1;
|
|
|
|
return !are_equivalent_nodes(a, b, InClass);
|
|
}
|