String Concat

Run Settings
LanguageC
Language Version
Run Command
#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; }
Editor Settings
Theme
Key bindings
Full width
Lines