2

我在 OSX 上发现了 vsprintf 的奇怪行为。

这是我的代码:

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#if defined(WIN32)
#include <windows.h>
#define VSNPRINTF _vsnprintf
#elif defined(LINUX) || defined (DARWIN)
#define VSNPRINTF vsnprintf
#include <sys/types.h>
#include <unistd.h>
#endif

char *f(const char *fmt, ...)
{
 char *out = NULL;
 const int step = 32;
 int n = -1, lout = step;
 va_list arg;

 if(fmt!=NULL)
 {
  va_start(arg, fmt);
  do
  {
   if(!out)
   {
    free(out);
    out = NULL;
   }
   out = (char*)malloc(lout + 1);
   if(!out) break;
   memset(out, 0, lout + 1);   

   n = VSNPRINTF(out, lout, fmt, arg);
   if(n == -1 || n + 1 > lout)
   {
    lout += step;
    n = -1;
   }
  }while(n == -1);
  va_end(arg);
 }

 return out;
}

int main()
{
 char *msg = NULL;
 unsigned long x = 0xDEADBEEF;

 msg = f("%X%X%X%X", x, x, x, x);
 if(!msg) return -1;

 puts(msg);

 return 0;
}

该函数应返回包含格式化文本的分配字符串 (char*)。它可以在 Linux 和 Windows 上正常工作。它在 OSX 上返回格式错误的文本,有时会导致分段错误 (EXC_BAD_ACCESS)。顺便说一句,我知道我可以使用 vasprintf。

可能是什么问题?

4

1 回答 1

5

您的问题很可能是您多次使用相同的 va_list 调用 vsnprintf 。这在某些 ABI 中不起作用。

查找 va_copy 的手册页。简短的版本是做这样的事情:

      va_list c;
      va_copy(c, arg);

      n = VSNPRINTF(out, lout, fmt, c);
      va_end(c);
于 2012-11-22T08:55:12.630 回答