Fix Python compile errors with overloading and varargs

Fixes wrapping overloaded functions/constructors where a vararg
function is declared after a non-vararg function.
This is a long standing bug in the Python layer exposed since fastunpack
was turned on by default.
This commit is contained in:
William S Fulton 2019-02-25 19:27:23 +00:00
parent 613ff08150
commit 83ea2280e2
11 changed files with 199 additions and 22 deletions

View File

@ -7,6 +7,10 @@ the issue number to the end of the URL: https://github.com/swig/swig/issues/
Version 4.0.0 (in progress)
===========================
2019-02-25: wsfulton
[Python] Fix compile errors wrapping overloaded functions/constructors where a vararg
function is declared after a non-vararg function.
2019-02-23: zphensley42
Use fully qualified name 'java.lang.Object' instead of 'Object' in generated code to
avoid clashes with wrapped C++ classes called 'Object'.

View File

@ -35,4 +35,40 @@ func main() {
if varargs_overload.Vararg_over4("Hello", 123) != "Hello" {
panic(8)
}
// Same as above but non-vararg function declared first
if varargs_overload.Vararg_over6("Hello") != "Hello" {
panic(0)
}
if varargs_overload.Vararg_over6(2) != "2" {
panic(1)
}
if varargs_overload.Vararg_over7("Hello") != "Hello" {
panic(2)
}
if varargs_overload.Vararg_over7(2, 2.2) != "2 2.2" {
panic(3)
}
if varargs_overload.Vararg_over8("Hello") != "Hello" {
panic(4)
}
if varargs_overload.Vararg_over8(2, 2.2, "hey") != "2 2.2 hey" {
panic(5)
}
if varargs_overload.Vararg_over9("Hello") != "Hello" {
panic(6)
}
if varargs_overload.Vararg_over9(123) != "123" {
panic(7)
}
if varargs_overload.Vararg_over9("Hello", 123) != "Hello" {
panic(8)
}
}

View File

@ -11,5 +11,19 @@ let _ =
assert (_vararg_over3 '(2, 2.2, "hey") as string = "2 2.2 hey");
assert (_vararg_over4 '("Hello") as string = "Hello");
assert (_vararg_over4 '(123) as string = "123");
assert (_vararg_over4 '("Hello", 123) as string = "Hello")
assert (_vararg_over4 '("Hello", 123) as string = "Hello");
(* Same as above but non-vararg function declared first *)
assert (_vararg_over6 '("Hello") as string = "Hello");
assert (_vararg_over6 '(2) as string = "2");
assert (_vararg_over7 '("Hello") as string = "Hello");
assert (_vararg_over7 '(2, 2.2) as string = "2 2.2");
assert (_vararg_over8 '("Hello") as string = "Hello");
assert (_vararg_over8 '(2, 2.2, "hey") as string = "2 2.2 hey");
assert (_vararg_over9 '("Hello") as string = "Hello");
assert (_vararg_over9 '(123) as string = "123");
assert (_vararg_over9 '("Hello", 123) as string = "Hello");
;;

View File

@ -28,3 +28,35 @@ if varargs_overload.vararg_over4(123) != "123":
if varargs_overload.vararg_over4("Hello", 123) != "Hello":
raise RuntimeError, "Failed"
# Same as above but non-vararg function declared first
if varargs_overload.vararg_over6("Hello") != "Hello":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over6(2) != "2":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over7("Hello") != "Hello":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over7(2, 2.2) != "2 2.2":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over8("Hello") != "Hello":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over8(2, 2.2, "hey") != "2 2.2 hey":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over9("Hello") != "Hello":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over9(123) != "123":
raise RuntimeError, "Failed"
if varargs_overload.vararg_over9("Hello", 123) != "Hello":
raise RuntimeError, "Failed"

View File

@ -4,7 +4,7 @@ checkequal(vararg_over1("Hello"), "Hello", "vararg_over1(""Hello"")");
checkequal(vararg_over1(2), "2", "vararg_over1(2)");
checkequal(vararg_over2("Hello"), "Hello", "vararg_over1(""Hello"")");
checkequal(vararg_over2("Hello"), "Hello", "vararg_over2(""Hello"")");
checkequal(vararg_over2(2, 2.2), "2 2.2", "vararg_over2(2, 2.2)")
@ -18,4 +18,24 @@ checkequal(vararg_over4(123), "123", "vararg_over4(123)");
checkequal(vararg_over4("Hello", 123), "Hello", "vararg_over4(""Hello"", 123)");
// Same as above but non-vararg function declared first
checkequal(vararg_over6("Hello"), "Hello", "vararg_over6(""Hello"")");
checkequal(vararg_over6(2), "2", "vararg_over6(2)");
checkequal(vararg_over7("Hello"), "Hello", "vararg_over7(""Hello"")");
checkequal(vararg_over7(2, 2.2), "2 2.2", "vararg_over7(2, 2.2)")
checkequal(vararg_over8("Hello"), "Hello", "vararg_over8(""Hello"")");
checkequal(vararg_over8(2, 2.2, "hey"), "2 2.2 hey", "vararg_over8(2, 2.2, ""hey"")");
checkequal(vararg_over9("Hello"), "Hello", "vararg_over9(""Hello"")");
checkequal(vararg_over9(123), "123", "vararg_over9(123)");
checkequal(vararg_over9("Hello", 123), "Hello", "vararg_over9(""Hello"", 123)");
exec("swigtest.quit", -1);

View File

@ -44,3 +44,47 @@ const char *vararg_over4(int i) {
return buffer;
}
%}
// Same as above but non-vararg function declared first
%inline %{
const char *vararg_over6(int i) {
static char buffer[256];
sprintf(buffer, "%d", i);
return buffer;
}
const char *vararg_over6(const char *fmt, ...) {
return fmt;
}
const char *vararg_over7(int i, double j) {
static char buffer[256];
sprintf(buffer, "%d %g", i, j);
return buffer;
}
const char *vararg_over7(const char *fmt, ...) {
return fmt;
}
const char *vararg_over8(int i, double j, const char *s) {
static char buffer[256];
sprintf(buffer, "%d %g %s", i, j, s);
return buffer;
}
const char *vararg_over8(const char *fmt, ...) {
return fmt;
}
%}
%varargs(int mode = 0) vararg_over9;
%inline %{
const char *vararg_over9(int i) {
static char buffer[256];
sprintf(buffer, "%d", i);
return buffer;
}
const char *vararg_over9(const char *fmt, ...) {
return fmt;
}
%}

View File

@ -188,7 +188,9 @@ void emit_attach_parmmaps(ParmList *l, Wrapper *f) {
p = lp;
while (p) {
if (SwigType_isvarargs(Getattr(p, "type"))) {
// Mark the head of the ParmList that it has varargs
Setattr(l, "emit:varargs", lp);
//Printf(stdout, "setting emit:varargs %s ... %s +++ %s\n", Getattr(l, "emit:varargs"), Getattr(l, "type"), Getattr(p, "type"));
break;
}
p = nextSibling(p);
@ -329,7 +331,8 @@ int emit_num_required(ParmList *parms) {
/* -----------------------------------------------------------------------------
* emit_isvarargs()
*
* Checks if a function is a varargs function
* Checks if a ParmList is a parameter list containing varargs.
* This function requires emit_attach_parmmaps to have been called beforehand.
* ----------------------------------------------------------------------------- */
int emit_isvarargs(ParmList *p) {
@ -340,6 +343,28 @@ int emit_isvarargs(ParmList *p) {
return 0;
}
/* -----------------------------------------------------------------------------
* emit_isvarargs_function()
*
* Checks for varargs in a function/constructor (can be overloaded)
* ----------------------------------------------------------------------------- */
bool emit_isvarargs_function(Node *n) {
bool has_varargs = false;
Node *over = Getattr(n, "sym:overloaded");
if (over) {
for (Node *sibling = over; sibling; sibling = Getattr(sibling, "sym:nextSibling")) {
if (ParmList_has_varargs(Getattr(sibling, "parms"))) {
has_varargs = true;
break;
}
}
} else {
has_varargs = ParmList_has_varargs(Getattr(n, "parms")) ? true : false;
}
return has_varargs;
}
/* -----------------------------------------------------------------------------
* void emit_mark_vararg_parms()
*

View File

@ -2187,7 +2187,7 @@ public:
* is_real_overloaded()
*
* Check if the function is overloaded, but not just have some
* siblings generated due to the original function have
* siblings generated due to the original function having
* default arguments.
* ------------------------------------------------------------ */
bool is_real_overloaded(Node *n) {
@ -2689,7 +2689,6 @@ public:
bool add_self = builtin_self && (!builtin_ctor || director_class);
bool builtin_getter = (builtin && GetFlag(n, "memberget"));
bool builtin_setter = (builtin && GetFlag(n, "memberset") && !builtin_getter);
bool over_varargs = false;
char const *self_param = builtin ? "self" : "SWIGUNUSEDPARM(self)";
char const *wrap_return = builtin_ctor ? "int " : "PyObject *";
String *linkage = NewString("SWIGINTERN ");
@ -2765,22 +2764,7 @@ public:
}
}
if (overname) {
String *over_varargs_attr = Getattr(n, "python:overvarargs");
if (!over_varargs_attr) {
for (Node *sibling = n; sibling; sibling = Getattr(sibling, "sym:nextSibling")) {
if (emit_isvarargs(Getattr(sibling, "parms"))) {
over_varargs = true;
break;
}
}
over_varargs_attr = NewString(over_varargs ? "1" : "0");
for (Node *sibling = n; sibling; sibling = Getattr(sibling, "sym:nextSibling"))
Setattr(sibling, "python:overvarargs", over_varargs_attr);
}
if (Strcmp(over_varargs_attr, "0") != 0)
over_varargs = true;
}
bool over_varargs = emit_isvarargs_function(n);
int funpack = fastunpack && !varargs && !over_varargs && !allow_kwargs;
int noargs = funpack && (tuple_required == 0 && tuple_arguments == 0);

View File

@ -383,7 +383,8 @@ List *SWIG_output_files();
void SWIG_library_directory(const char *);
int emit_num_arguments(ParmList *);
int emit_num_required(ParmList *);
int emit_isvarargs(ParmList *);
int emit_isvarargs(ParmList *p);
bool emit_isvarargs_function(Node *n);
void emit_attach_parmmaps(ParmList *, Wrapper *f);
void emit_mark_varargs(ParmList *l);
String *emit_action(Node *n);

View File

@ -254,3 +254,19 @@ int ParmList_has_defaultargs(ParmList *p) {
}
return 0;
}
/* ---------------------------------------------------------------------
* ParmList_has_varargs()
*
* Returns 1 if the parameter list passed in has varargs.
* Otherwise returns 0.
* ---------------------------------------------------------------------- */
int ParmList_has_varargs(ParmList *p) {
Parm *lp = 0;
while (p) {
lp = p;
p = nextSibling(p);
}
return lp ? SwigType_isvarargs(Getattr(lp, "type")) : 0;
}

View File

@ -24,6 +24,7 @@ extern ParmList *CopyParmListMax(ParmList *, int count);
extern int ParmList_len(ParmList *);
extern int ParmList_numrequired(ParmList *);
extern int ParmList_has_defaultargs(ParmList *p);
extern int ParmList_has_varargs(ParmList *p);
/* Output functions */
extern String *ParmList_str(ParmList *);