最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

"Extensible" C generics - Stack Overflow

programmeradmin2浏览0评论

I'm trying to make an implementation of a Vector in C (yes, I know, this has been done to death), generic across any types used. So far I have an implementation something like:

#define Vector(T) struct Vector_##T

// other routines, initialization, destruction, ...
// an example routine
#define DEFINE_VectorFirst(T)                                                  \
  T VectorFirst_##T(Vector(T) vec) {                                           \
    return vec.data[0];                                                        \
  }
#define DEFINE_VECTOR(T)                                                       \
  struct Vector_##T {                                                          \
    T *data;                                                                   \
    size_t len;                                                                \
    size_t capacity;                                                           \
  };                                                                           \
  DEFINE_VectorFirst(T)

// a few types for now
DEFINE_VECTOR(int)
DEFINE_VECTOR(double)

so I can, for example, declare a Vector(double) and then call VectorFirst_double on it. Ideally I'd like to not have to specify the type name every time, so I can use a generic:

#define GENERIC(F, T) Vector(T) : F##_##T
#define VectorFirst(v)                                                         \
  _Generic((v), GENERIC(VectorFirst, int), GENERIC(VectorFirst, double))(v)

and now VectorFirst automatically calls the right version, but this loses the nice property that I can DEFINE_VECTOR in a bunch of different header files. In this way, the original version is "extensible", letting me make vectors of whatever structs I might declare elsewhere, but in the generic one I need to know all of the vector types I might ever need at one spot. I could just use macros instead of functions for some things (#define VectorFirst(v) v.data[0] or the like) but it seems like this would make debugging annoying and maybe be slow for larger functions. Are there any preprocessor techniques that can be used to restore this extensibility behavior and let me add new entries to a _Generic somewhere else? This is a small-scale test so I'm okay with whatever awful preprocessor hacks people know of.

I'm trying to make an implementation of a Vector in C (yes, I know, this has been done to death), generic across any types used. So far I have an implementation something like:

#define Vector(T) struct Vector_##T

// other routines, initialization, destruction, ...
// an example routine
#define DEFINE_VectorFirst(T)                                                  \
  T VectorFirst_##T(Vector(T) vec) {                                           \
    return vec.data[0];                                                        \
  }
#define DEFINE_VECTOR(T)                                                       \
  struct Vector_##T {                                                          \
    T *data;                                                                   \
    size_t len;                                                                \
    size_t capacity;                                                           \
  };                                                                           \
  DEFINE_VectorFirst(T)

// a few types for now
DEFINE_VECTOR(int)
DEFINE_VECTOR(double)

so I can, for example, declare a Vector(double) and then call VectorFirst_double on it. Ideally I'd like to not have to specify the type name every time, so I can use a generic:

#define GENERIC(F, T) Vector(T) : F##_##T
#define VectorFirst(v)                                                         \
  _Generic((v), GENERIC(VectorFirst, int), GENERIC(VectorFirst, double))(v)

and now VectorFirst automatically calls the right version, but this loses the nice property that I can DEFINE_VECTOR in a bunch of different header files. In this way, the original version is "extensible", letting me make vectors of whatever structs I might declare elsewhere, but in the generic one I need to know all of the vector types I might ever need at one spot. I could just use macros instead of functions for some things (#define VectorFirst(v) v.data[0] or the like) but it seems like this would make debugging annoying and maybe be slow for larger functions. Are there any preprocessor techniques that can be used to restore this extensibility behavior and let me add new entries to a _Generic somewhere else? This is a small-scale test so I'm okay with whatever awful preprocessor hacks people know of.

Share Improve this question asked 2 days ago Josh OpJosh Op 1356 bronze badges 9
  • 1 "across any types" --> I suspect #define GENERIC(F, T) Vector(T) : F##_##T will have trouble with type long long (types with a space in them). Your 1st approach apparently does not have that issue. – chux Commented 2 days ago
  • 1 ... which could be worked around by use of typedefs, but that's still a fairly nasty gotcha. – John Bollinger Commented 2 days ago
  • 1 Pointer types like int* deserve testing too. (@JohnBollinger and hiding a pointer type in a typedef is an antipattern) – chux Commented 2 days ago
  • @Josh Op, I think I would go the with a predefined set of types rather than any type. To fully handle any type you may have to do something like what Stroustrup did. – chux Commented 2 days ago
  • Presumably you use C because you don't want the compiler to generate stuff for you, or have different functions use the same name (overloads). Trying to force that into the program anyway, using generic macros, is just going to make you cry. Like other comments say, Bjarne tried that 40-50 years ago, and soon gave up. It doesn't work. – BoP Commented 2 days ago
 |  Show 4 more comments

1 Answer 1

Reset to default 3

Are there any preprocessor techniques that can be used to restore this extensibility behavior and let me add new entries to a _Generic somewhere else?

Not as such. But you have to understand that _Generic is a (weird) operator. Every appearance is separate, including every instance resulting from expansion of a macro. There is no meaningful sense of "elsewhere" in which one could add entries to a particular generic selection expression.

Macros, on the other hand, can be defined and undefined fairly freely. You don't have to have a common definition that is shared everywhere. "Somewhere else" can provide their own definition that suits them.

You could perhaps facilitate each translation unit defining a suitable set of type-generic macros by leveraging X macros. It might look something like this:

generic_vector.h

// ...

// A suitable GENERIC_OPTS definition must be in scope already
#ifndef GENERIC_OPTS
    #error no definition of GENERIC_OPTS
#endif

#define GENERIC(F, T) Vector(T) : F##_##T

#define VF_DECL(T) , GENERIC(VectorFirst, T)
#define VectorFirst(v)       \
  _Generic((v)               \
    GENERIC_OPTS(VF_DECL)    \
  )(v)

// ... other operations ...

other.c

#define GENERIC_OPTS(G) \
  G(int) \
  G(double) \
  G(uint64_t)

#include "generic_vector.h"

#undef GENERIC_OPTS
// ...

There, the definition of GENERIC_OPTS in other.c controls what element types can be used with the VectorFirst() macro as defined in that file. If you need to generate function definitions as well, then they can be accommodated the same way, but you will want to declare them with internal linkage (i.e. static) if each translation unit is defining its own.


Do be aware, however, that C data type names can be arbitrarily complex. Even among just the (unqualified) standard arithmetic types, a majority have multi-word names, as @chux alluded to in comments. Your GENERIC() macro for building type-specific identifiers from type names will work as you intend only for type names that are single identifiers.

You can work around that by defining typedefs for types whose names are not a single identifier, but that gets nasty fast. If you really want to pursue this idea, then you should consider writing a full-fledged code generator instead of relying on the limited capabilities of the C preprocessor.

发布评论

评论列表(0)

  1. 暂无评论