#include <stdio.h>
#include <stdarg.h>
#include <stdlib.h>
#include <stdint.h>
#include <assert.h>
#include <string.h>
#define VA_NUM_ARGS_IMPL(_1,_2,_3,_4,_5,_6,_7,_8,_9,_10,_11,_12,_13,_14, \
_15,_16,_17,_18,_19,_20,_21,_22,_23,_24,_25,_26,_27,_28,_29,_30,_31,_32, \
_33,_34,_35,_36,_37,_38,_39,_40,_41,_42,_43,_44,_45,_46,_47,_48,_49,_50, \
_51,_52,_53,_54,_55,_56,_57,_58,_59,_60,_61,_62,_63,_64,_65,_66,_67,_68, \
_69,_70,_71,_72,_73,_74,_75,_76,_77,_78,_79,_80,_81,_82,_83,_84,_85,_86, \
_87,_88,_89,_90,_91,_92,_93,_94,_95,_96,_97,_98,_99,_100,_101,_102,_103, \
_104,_105,_106,_107,_108,_109,_110,_111,_112,_113,_114,_115,_116,_117, \
_118,_119,_120,_121,_122,_123,_124,_125,_126,_127,_128,_129,_130,_131, \
_132,_133,_134,_135,_136,_137,_138,_139,_140,_141,_142,_143,_144,_145, \
_146,_147,_148,_149,_150,_151,_152,_153,_154,_155,_156,_157,_158,_159, \
_160,_161,_162,_163,_164,_165,_166,_167,_168,_169,_170,_171,_172,_173, \
_174,_175,_176,_177,_178,_179,_180,_181,_182,_183,_184,_185,_186,_187, \
_188,_189,_190,_191,_192,_193,_194,_195,_196,_197,_198,_199,_200,_201, \
_202,_203,_204,_205,_206,_207,_208,_209,_210,_211,_212,_213,_214,_215, \
_216,_217,_218,_219,_220,_221,_222,_223,_224,_225,_226,_227,_228,_229, \
_230,_231,_232,_233,_234,_235,_236,_237,_238,_239,_240,_241,_242,_243, \
_244,_245,_246,_247,_248,_249,_250,_251,_252,_253,_254,_255,N,...) N
#define __VA_ARG_COUNT__(...) (sizeof(#__VA_ARGS__) == sizeof("") ? 0 : \
VA_NUM_ARGS_IMPL(__VA_ARGS__,255,254,253,252,251,250,249,248,247,246,245, \
244,243,242,241,240,239,238,237,236,235,234,233,232,231,230,229,228,227, \
226,225,224,223,222,221,220,219,218,217,216,215,214,213,212,211,210,209, \
208,207,206,205,204,203,202,201,200,199,198,197,196,195,194,193,192,191, \
190,189,188,187,186,185,184,183,182,181,180,179,178,177,176,175,174,173, \
172,171,170,169,168,167,166,165,164,163,162,161,160,159,158,157,156,155, \
154,153,152,151,150,149,148,147,146,145,144,143,142,141,140,139,138,137, \
136,135,134,133,132,131,130,129,128,127,126,125,124,123,122,121,120,119, \
118,117,116,115,114,113,112,111,110,109,108,107,106,105,104,103,102,101, \
100,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77, \
76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53, \
52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29, \
28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,1))
//varcall(...) is a setup for call-through. So the first argument is
//the number of arguments, then arguments.
#define varcall(...) __VA_ARG_COUNT__(__VA_ARGS__), ##__VA_ARGS__
//varfunc() takes a function as the first argument then the vararg list.
//func() will be called with the count of arguments as its first parameter,
//then the varargs.
#define varfunc(func, ...) func(varcall(__VA_ARGS__))
//Note that varfunc have to used at call sight and varcall have to be used
//in the definition of a function.
//Ergo: No clean way to handle varargs.
typedef struct{
int32_t Size;
int32_t Index;
char *Buffer;
}Buffer;
Buffer BufferInit(int Size)
{
Buffer Buf;
Buf.Size = Size;
Buf.Index = 0;
Buf.Buffer = (char *)malloc(Buf.Size);
assert(Buf.Buffer);
memset(Buf.Buffer, 0, Buf.Size);
return Buf;
}
void Buffer_Resize(Buffer *Buf)
{
Buf->Size *= 2;
Buf->Buffer = realloc(Buf->Buffer, Buf->Size);
assert(Buf->Buffer);
}
void Buffer_Add(Buffer *String, char Character)
{
String->Buffer[String->Index++] = Character;
if(String->Index > String->Size - 1)
Buffer_Resize(String);
}
#define VarInit \
va_list Uninteresting_Arg_Handle_; \
va_start(Uninteresting_Arg_Handle_, VarCount)
#define VarDeinit \
va_end(Uninteresting_Arg_Handle_)
#define VarGrab(Type) \
va_arg(Uninteresting_Arg_Handle_, Type); \
{ --VarCount; assert(VarCount > -1); }
#define Concat(...) Concat_(varcall(__VA_ARGS__))
void Concat_(int VarCount, ...)
{
VarInit;
Buffer *OutString = VarGrab(Buffer *);
assert(OutString);
while(VarCount)
{
char *String = VarGrab(char *);
int j = 0;
while(String[j])
{
Buffer_Add(OutString, String[j]);
++j;
}
}
VarDeinit;
}
int main()
{
Buffer String = BufferInit(20);
Concat(&String, "Nice ", "demo ", "of\n");
Concat(&String, "variable\n");
Concat(&String);
varfunc(Concat_, &String, "string ", "arguments\n");
printf(String.Buffer);
return 0;
}