cstdio的源代码

cstdio是将stdio.h的内容用C++头文件的形式表示出来。stdio.h是C标准函数库中的头文,即:standard buffered input&output。提供基本的文字的输入输出流操作(包括屏幕和文件等)。由于C语言并没有提供专用于文字输入输出的关键字,所以该库是最普遍的C语言程序加载库。

// -*- C++ -*- forwarding header.

// Copyright (C) 1997-2019 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library.  This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.

// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// Under Section 7 of GPL version 3, you are granted additional
// permissions described in the GCC Runtime Library Exception, version
// 3.1, as published by the Free Software Foundation.

// You should have received a copy of the GNU General Public License and
// a copy of the GCC Runtime Library Exception along with this program;
// see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
// <http://www.gnu.org/licenses/>.

/** @file include/cstdio
 *  This is a Standard C++ Library file.  You should @c \#include this file
 *  in your programs, rather than any of the @a *.h implementation files.
 *
 *  This is the C++ version of the Standard C Library header @c stdio.h,
 *  and its contents are (mostly) the same as that header, but are all
 *  contained in the namespace @c std (except for names which are defined
 *  as macros in C).
 */

//
// ISO C++ 14882: 27.8.2  C Library files
//

#pragma GCC system_header

#include <bits/c++config.h>
#include <stdio.h>

#ifndef _GLIBCXX_CSTDIO
#define _GLIBCXX_CSTDIO 1

#if __cplusplus <= 201103L && !defined(_GLIBCXX_HAVE_GETS)
extern "C" char* gets (char* __s) __attribute__((__deprecated__));
#endif

// Get rid of those macros defined in <stdio.h> in lieu of real functions.
#undef clearerr
#undef fclose
#undef feof
#undef ferror
#undef fflush
#undef fgetc
#undef fgetpos
#undef fgets
#undef fopen
#undef fprintf
#undef fputc
#undef fputs
#undef fread
#undef freopen
#undef fscanf
#undef fseek
#undef fsetpos
#undef ftell
#undef fwrite
#undef getc
#undef getchar
#if __cplusplus <= 201103L
# undef gets
#endif
#undef perror
#undef printf
#undef putc
#undef putchar
#undef puts
#undef remove
#undef rename
#undef rewind
#undef scanf
#undef setbuf
#undef setvbuf
#undef sprintf
#undef sscanf
#undef tmpfile
#undef tmpnam
#undef ungetc
#undef vfprintf
#undef vprintf
#undef vsprintf

namespace std
{
  using ::FILE;
  using ::fpos_t;

  using ::clearerr;
  using ::fclose;
  using ::feof;
  using ::ferror;
  using ::fflush;
  using ::fgetc;
  using ::fgetpos;
  using ::fgets;
  using ::fopen;
  using ::fprintf;
  using ::fputc;
  using ::fputs;
  using ::fread;
  using ::freopen;
  using ::fscanf;
  using ::fseek;
  using ::fsetpos;
  using ::ftell;
  using ::fwrite;
  using ::getc;
  using ::getchar;
#if __cplusplus <= 201103L
  // LWG 2249
  using ::gets;
#endif
  using ::perror;
  using ::printf;
  using ::putc;
  using ::putchar;
  using ::puts;
  using ::remove;
  using ::rename;
  using ::rewind;
  using ::scanf;
  using ::setbuf;
  using ::setvbuf;
  using ::sprintf;
  using ::sscanf;
  using ::tmpfile;
#if _GLIBCXX_USE_TMPNAM
  using ::tmpnam;
#endif
  using ::ungetc;
  using ::vfprintf;
  using ::vprintf;
  using ::vsprintf;
} // namespace

#if _GLIBCXX_USE_C99_STDIO

#undef snprintf
#undef vfscanf
#undef vscanf
#undef vsnprintf
#undef vsscanf

namespace __gnu_cxx
{
#if _GLIBCXX_USE_C99_CHECK || _GLIBCXX_USE_C99_DYNAMIC
  extern "C" int
  (snprintf)(char * __restrict, std::size_t, const char * __restrict, ...)
  throw ();
  extern "C" int
  (vfscanf)(FILE * __restrict, const char * __restrict, __gnuc_va_list);
  extern "C" int (vscanf)(const char * __restrict, __gnuc_va_list);
  extern "C" int
  (vsnprintf)(char * __restrict, std::size_t, const char * __restrict,
	      __gnuc_va_list) throw ();
  extern "C" int
  (vsscanf)(const char * __restrict, const char * __restrict, __gnuc_va_list)
  throw ();
#endif

#if !_GLIBCXX_USE_C99_DYNAMIC
  using ::snprintf;
  using ::vfscanf;
  using ::vscanf;
  using ::vsnprintf;
  using ::vsscanf;
#endif
} // namespace __gnu_cxx

namespace std
{
  using ::__gnu_cxx::snprintf;
  using ::__gnu_cxx::vfscanf;
  using ::__gnu_cxx::vscanf;
  using ::__gnu_cxx::vsnprintf;
  using ::__gnu_cxx::vsscanf;
} // namespace std

#endif // _GLIBCXX_USE_C99_STDIO

#endif

有问题的话欢迎在聊天区里留言哦~

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果需要打印堆栈信息中对应的代码位置,可以使用`addr2line`命令来实现。`addr2line`命令可以将一个地址转换为对应的源代码文件名和行号。 ```c++ #include <execinfo.h> #include <cxxabi.h> #include <cstdio> #include <cstdlib> #include <cstring> #include <dlfcn.h> #include <unistd.h> #include <fcntl.h> #include <sys/wait.h> void printStackTrace() { void* trace[256]; int traceSize = backtrace(trace, 256); char** messages = backtrace_symbols(trace, traceSize); int pid = getpid(); char procSelfExe[256]; sprintf(procSelfExe, "/proc/%d/exe", pid); char exePath[256]; int exePathSize = readlink(procSelfExe, exePath, sizeof(exePath)); exePath[exePathSize] = '\0'; int addr2lineIn = open("/tmp/addr2lineIn", O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR); int addr2lineOut = open("/tmp/addr2lineOut", O_RDONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR); for (int i = 1; i < traceSize; ++i) { Dl_info info; dladdr(trace[i], &info); char* demangled = nullptr; int status = 0; demangled = abi::__cxa_demangle(info.dli_sname, nullptr, 0, &status); printf("[bt]: (%d) %s (%s + 0x%lx)\n", i, messages[i], (status==0 ? demangled : info.dli_sname), (char*)trace[i] - (char*)info.dli_saddr); uint64_t addr = (uint64_t)trace[i]; write(addr2lineIn, &addr, sizeof(addr)); } pid_t child = fork(); if (child == 0) { dup2(addr2lineIn, STDIN_FILENO); dup2(addr2lineOut, STDOUT_FILENO); execlp("addr2line", "addr2line", "-e", exePath, "-f", "-i", nullptr); } waitpid(child, nullptr, 0); lseek(addr2lineOut, 0, SEEK_SET); char buf[256]; for (int i = 1; i < traceSize; ++i) { uint64_t addr; read(addr2lineOut, &addr, sizeof(addr)); if (addr == 0) { printf("[bt]: ??:0\n"); continue; } fgets(buf, sizeof(buf), addr2lineOut); char* p = strchr(buf, '\n'); if (p != nullptr) { *p = '\0'; } printf("[bt]: %s\n", buf); } close(addr2lineIn); close(addr2lineOut); free(messages); } ``` 上述代码中,我们使用`/proc/self/exe`获取当前进程的可执行文件路径,并通过`addr2line`命令将地址转换为代码位置。具体实现过程为: 1. 将地址写入`/tmp/addr2lineIn`文件,`addr2line`命令从该文件中读取地址。 2. 使用`fork()`函数创建子进程,重定向子进程的标准输入和标准输出到`/tmp/addr2lineIn`和`/tmp/addr2lineOut`文件,使`addr2line`命令可以从标准输入中读取地址,并将结果输出到标准输出中。 3. 在子进程中使用`execlp()`函数调用`addr2line`命令,并传入可执行文件路径、`-f`和`-i`选项,表示输出完整的函数名和文件名、输出源码文件名和行号。 4. 等待子进程执行完毕,再从`/tmp/addr2lineOut`文件中读取`addr2line`命令的输出结果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值