Friend operator overloading fix for Python builtin

Don't generate calls to
  SWIGPY_BINARYFUNC_CLOSURE(_wrap___lshift__)
for the 'global' function wrappers for the friend functions.

Only occurred when a friend class is within a namespace due to the
renames in pyopers.swg, such as:
  %rename(__lshift__) *::operator<<;
The rename was intended for member functions but was also being attached
to friends that are within a namespace.
This commit is contained in:
William S Fulton 2024-01-27 14:41:23 +00:00
parent 521d43d071
commit 344ca5fbc6
5 changed files with 162 additions and 1 deletions

View File

@ -7,6 +7,10 @@ the issue number to the end of the URL: https://github.com/swig/swig/issues/
Version 4.2.1 (in progress)
===========================
2024-01-27: wsfulton
[Python] Fix compilation error when wrapping two or more classes that
have the same friend operator overload when the classes are in a namespace.
2024-01-15: wsfulton
https://sourceforge.net/p/swig/bugs/960/
https://sourceforge.net/p/swig/bugs/807/

View File

@ -263,6 +263,7 @@ CPP_TEST_CASES += \
features \
fragments \
friends \
friends_operator_overloading \
friends_template \
funcptr_cpp \
functors \

View File

@ -0,0 +1,114 @@
%module friends_operator_overloading
// Tests friend operators within a namespace
// Demonstrates how to turn friend operators into member functions (required for some languages - tests includes a Python runtime test)
// Note that by default the friend functions result in global function wrappers (overloaded as there are friends from two different classes)
// Testcase highlighted a compilation problem with Python builtin wrappers
// Tests only the languages that don't ignore operator<<
%warnfilter(SWIGWARN_LANG_IDENTIFIER, // Warning 503: Can't wrap 'operator <<' unless renamed to a valid identifier.
SWIGWARN_IGNORE_OPERATOR_LSHIFT) operator<<; // Warning 373: operator<< ignored
%inline %{
// Remove this define to test the equivalent implementation using member methods instead of friends
#define FRIENDS
// Debugging/tracing using printf
#include <cstdio>
//#define myprintf(a, b) printf(a, b)
#define myprintf(a, b)
namespace shifting {
class ShiftA {
int val;
public:
ShiftA(int val = 0) : val(val) {}
#if !defined(FRIENDS)
ShiftA operator<<(const ShiftA& gd) {
ShiftA ret(val - gd.getVal());
myprintf("member operator << (GeoData) %d\n", ret.getVal());
return ret;
}
ShiftA operator<<(int amount) {
ShiftA ret(val - amount);
myprintf("member operator << (int) %d\n", ret.getVal());
return ret;
}
#else
friend ShiftA operator<<(const ShiftA& this_, const ShiftA& gd) {
ShiftA ret(this_.val - gd.getVal());
myprintf("friend operator << (GeoData) %d\n", ret.getVal());
return ret;
}
friend ShiftA operator<<(const ShiftA& this_, int amount) {
ShiftA ret(this_.val - amount);
myprintf("friend operator << (int) %d\n", ret.getVal());
return ret;
}
#if defined(SWIG)
%extend {
ShiftA operator<<(const ShiftA& gd) { return *$self << gd; }
ShiftA operator<<(int amount) { return *$self << amount; }
}
#endif
#endif
int getVal() const { return val; }
};
class ShiftB {
int val;
public:
ShiftB(int val = 0) : val(val) {}
#if !defined(FRIENDS)
ShiftB operator<<(const ShiftB& gd) {
ShiftB ret(val - gd.getVal());
myprintf("member operator << (GeoData) %d\n", ret.getVal());
return ret;
}
ShiftB operator<<(int amount) {
ShiftB ret(val - amount);
myprintf("member operator << (int) %d\n", ret.getVal());
return ret;
}
#else
friend ShiftB operator<<(const ShiftB& this_, const ShiftB& gd) {
ShiftB ret(this_.val - gd.getVal());
myprintf("friend operator << (GeoData) %d\n", ret.getVal());
return ret;
}
friend ShiftB operator<<(const ShiftB& this_, int amount) {
ShiftB ret(this_.val - amount);
myprintf("friend operator << (int) %d\n", ret.getVal());
return ret;
}
#if defined(SWIG)
%extend {
ShiftB operator<<(const ShiftB& gd) { return *$self << gd; }
ShiftB operator<<(int amount) { return *$self << amount; }
}
#endif
#endif
int getVal() const { return val; }
};
void sanity_checker_ShiftA() {
ShiftA gd1(20);
ShiftA gd2(100);
ShiftA gd3(gd2 << gd1);
myprintf("gd3 %d\n", gd3.getVal());
ShiftA gd4(gd2 << 30);
myprintf("gd4 %d\n", gd4.getVal());
}
void sanity_checker_ShiftB() {
ShiftB gd1(20);
ShiftB gd2(100);
ShiftB gd3(gd2 << gd1);
myprintf("gd3 %d\n", gd3.getVal());
ShiftB gd4(gd2 << 30);
myprintf("gd4 %d\n", gd4.getVal());
}
}
%}

View File

@ -0,0 +1,41 @@
import friends_operator_overloading
friends_operator_overloading.sanity_checker_ShiftA()
friends_operator_overloading.sanity_checker_ShiftB()
sa1 = friends_operator_overloading.ShiftA(200)
sa2 = friends_operator_overloading.ShiftA(1000)
sb1 = friends_operator_overloading.ShiftB(200)
sb2 = friends_operator_overloading.ShiftB(1000)
# Shift operator via members
sa3 = sa2 << sa1
val = sa3.getVal()
if val != 800:
raise RuntimeError("Wrong val: {}".format(val))
sa4 = sa2 << 300
val = sa4.getVal()
if val != 700:
raise RuntimeError("Wrong val: {}".format(val))
sb3 = sb2 << sb1
val = sb3.getVal()
if val != 800:
raise RuntimeError("Wrong val: {}".format(val))
sb4 = sb2 << 300
val = sb4.getVal()
if val != 700:
raise RuntimeError("Wrong val: {}".format(val))
# Shift operator via global wrapper
shift = friends_operator_overloading.__lshift__(sa2, sa1)
val = shift.getVal()
if val != 800:
raise RuntimeError("Wrong val: {}".format(val))
shift = friends_operator_overloading.__lshift__(sb2, sb1)
val = shift.getVal()
if val != 800:
raise RuntimeError("Wrong val: {}".format(val))

View File

@ -2718,6 +2718,7 @@ public:
int constructor = (!Cmp(nodeType, "constructor"));
int destructor = (!Cmp(nodeType, "destructor"));
String *storage = Getattr(n, "storage");
int isfriend = Strstr(storage, "friend") != NULL;
/* Only the first constructor is handled as init method. Others
constructor can be emitted via %rename */
int handled_as_init = 0;
@ -3374,7 +3375,7 @@ public:
if (in_class && builtin) {
/* Handle operator overloads for builtin types */
String *slot = Getattr(n, "feature:python:slot");
if (slot) {
if (slot && !isfriend) {
String *func_type = Getattr(n, "feature:python:slot:functype");
String *closure_decl = getClosure(func_type, wrapper_name, overname ? 0 : funpack);
String *feature_name = NewStringf("feature:python:%s", slot);