mirror of https://github.com/swig/swig
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:
parent
521d43d071
commit
344ca5fbc6
|
@ -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)
|
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
|
2024-01-15: wsfulton
|
||||||
https://sourceforge.net/p/swig/bugs/960/
|
https://sourceforge.net/p/swig/bugs/960/
|
||||||
https://sourceforge.net/p/swig/bugs/807/
|
https://sourceforge.net/p/swig/bugs/807/
|
||||||
|
|
|
@ -263,6 +263,7 @@ CPP_TEST_CASES += \
|
||||||
features \
|
features \
|
||||||
fragments \
|
fragments \
|
||||||
friends \
|
friends \
|
||||||
|
friends_operator_overloading \
|
||||||
friends_template \
|
friends_template \
|
||||||
funcptr_cpp \
|
funcptr_cpp \
|
||||||
functors \
|
functors \
|
||||||
|
|
|
@ -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());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
%}
|
|
@ -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))
|
|
@ -2718,6 +2718,7 @@ public:
|
||||||
int constructor = (!Cmp(nodeType, "constructor"));
|
int constructor = (!Cmp(nodeType, "constructor"));
|
||||||
int destructor = (!Cmp(nodeType, "destructor"));
|
int destructor = (!Cmp(nodeType, "destructor"));
|
||||||
String *storage = Getattr(n, "storage");
|
String *storage = Getattr(n, "storage");
|
||||||
|
int isfriend = Strstr(storage, "friend") != NULL;
|
||||||
/* Only the first constructor is handled as init method. Others
|
/* Only the first constructor is handled as init method. Others
|
||||||
constructor can be emitted via %rename */
|
constructor can be emitted via %rename */
|
||||||
int handled_as_init = 0;
|
int handled_as_init = 0;
|
||||||
|
@ -3374,7 +3375,7 @@ public:
|
||||||
if (in_class && builtin) {
|
if (in_class && builtin) {
|
||||||
/* Handle operator overloads for builtin types */
|
/* Handle operator overloads for builtin types */
|
||||||
String *slot = Getattr(n, "feature:python:slot");
|
String *slot = Getattr(n, "feature:python:slot");
|
||||||
if (slot) {
|
if (slot && !isfriend) {
|
||||||
String *func_type = Getattr(n, "feature:python:slot:functype");
|
String *func_type = Getattr(n, "feature:python:slot:functype");
|
||||||
String *closure_decl = getClosure(func_type, wrapper_name, overname ? 0 : funpack);
|
String *closure_decl = getClosure(func_type, wrapper_name, overname ? 0 : funpack);
|
||||||
String *feature_name = NewStringf("feature:python:%s", slot);
|
String *feature_name = NewStringf("feature:python:%s", slot);
|
||||||
|
|
Loading…
Reference in New Issue