Skip to content

Commit

Permalink
py/obj.h: Fix compilation with C++
Browse files Browse the repository at this point in the history
Fixes issue micropython#12951.

Signed-off-by: stijn <[email protected]>
  • Loading branch information
stinos committed Nov 13, 2023
1 parent 05cb140 commit 6ccb266
Show file tree
Hide file tree
Showing 3 changed files with 19 additions and 7 deletions.
7 changes: 7 additions & 0 deletions examples/usercmodule/cppexample/example.cpp
Original file line number Diff line number Diff line change
@@ -1,9 +1,16 @@
extern "C" {
#include <examplemodule.h>
#include <py/objstr.h>

// Here we implement the function using C++ code, but since it's
// declaration has to be compatible with C everything goes in extern "C" scope.
mp_obj_t cppfunc(mp_obj_t a_obj, mp_obj_t b_obj) {
//The following no-ops are just here to verify the static assertions used in
//the public API all compile with C++.
MP_STATIC_ASSERT_STR_ARRAY_COMPATIBLE;
if (mp_obj_is_type(a_obj, &mp_type_BaseException)) {
}

// Prove we have (at least) C++11 features.
const auto a = mp_obj_get_int(a_obj);
const auto b = mp_obj_get_int(b_obj);
Expand Down
11 changes: 8 additions & 3 deletions py/misc.h
Original file line number Diff line number Diff line change
Expand Up @@ -52,10 +52,15 @@ typedef unsigned int uint;

// Static assertion macro
#define MP_STATIC_ASSERT(cond) ((void)sizeof(char[1 - 2 * !(cond)]))
#if defined(_MSC_VER)
#define MP_STATIC_ASSERT_NOT_MSC(cond) (1)
// In C++ things like comparing extern const pointers are not constant-expressions so cannot be used
// in MP_STATIC_ASSERT. Note that not all possible compiler versions will reject this. Some gcc versions
// do, others only with -Werror=vla, msvc always does.
// The (void) is needed to avoid "left operand of comma operator has no effect [-Werror=unused-value]"
// when using this macro on the left-hand side of a comma.
#if defined(__cplusplus)
#define MP_STATIC_ASSERT_IF_C(cond) ((void)1)
#else
#define MP_STATIC_ASSERT_NOT_MSC(cond) MP_STATIC_ASSERT(cond)
#define MP_STATIC_ASSERT_IF_C(cond) MP_STATIC_ASSERT(cond)
#endif

// Round-up integer division
Expand Down
8 changes: 4 additions & 4 deletions py/obj.h
Original file line number Diff line number Diff line change
Expand Up @@ -919,10 +919,10 @@ void *mp_obj_malloc_helper(size_t num_bytes, const mp_obj_type_t *type);
// optimizations (other tricks like using ({ expr; exper; }) or (exp, expr, expr) in mp_obj_is_type() result
// in missed optimizations)
#define mp_type_assert_not_bool_int_str_nonetype(t) ( \
MP_STATIC_ASSERT_NOT_MSC((t) != &mp_type_bool), assert((t) != &mp_type_bool), \
MP_STATIC_ASSERT_NOT_MSC((t) != &mp_type_int), assert((t) != &mp_type_int), \
MP_STATIC_ASSERT_NOT_MSC((t) != &mp_type_str), assert((t) != &mp_type_str), \
MP_STATIC_ASSERT_NOT_MSC((t) != &mp_type_NoneType), assert((t) != &mp_type_NoneType), \
MP_STATIC_ASSERT_IF_C((t) != &mp_type_bool), assert((t) != &mp_type_bool), \
MP_STATIC_ASSERT_IF_C((t) != &mp_type_int), assert((t) != &mp_type_int), \
MP_STATIC_ASSERT_IF_C((t) != &mp_type_str), assert((t) != &mp_type_str), \
MP_STATIC_ASSERT_IF_C((t) != &mp_type_NoneType), assert((t) != &mp_type_NoneType), \
1)

#define mp_obj_is_type(o, t) (mp_type_assert_not_bool_int_str_nonetype(t) && mp_obj_is_exact_type(o, t))
Expand Down

0 comments on commit 6ccb266

Please sign in to comment.