fdb7d390dd
This patch added a new fmemopen version, for glibc 2.22, that aims to be POSIX complaint. It fixes some long-stading glibc fmemopen issues, such as: * it changes the way fseek with SEEK_END works on fmemopen to seek relative to buffer size instead of first '\0'. This is default mode and 'b' opening mode does not change internal behavior (bz#6544). * fix apending opening mode to use as start position either first null byte of len specified in function call (bz#13152 and #13151). * remove binary option 'b' and internal different handling (bz#12836) * fix seek/SEE_END with negative values (bz#14292). A compatibility symbol is provided to with old behavior for older symbols version (2.2.5). * include/stdio.h (fmemopen): Remove hidden prototype. (__fmemopen): Add new hidden prototype. * libio/Makefile: Add oldfmemopen object. * libio/Versions [GLIBC_2.22]: Add new fmemopen symbol. * libio/fmemopen.c (__fmemopen): Function rewrite to be POSIX compliance. * libio/oldfmemopen.c: New file: old fmemopen implementation for symbol compatibility. * stdio-common/Makefile [tests]: Add new tst-fmemopen3. * stdio-common/psiginfo.c [psiginfo]: Call __fmemopen instead of fmemopen. * stdio-common/tst-fmemopen3.c: New file: more fmemopen tests, focus on append and read mode. * sysdeps/unix/sysv/linux/aarch64/libc.abilist [GLIBC_2.22]: Add fmemopen. * sysdeps/unix/sysv/linux/alpha/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/arm/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/i386/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/ia64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/m68k/coldfire/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/m68k/m680x0/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/microblaze/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/mips/mips32/fpu/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/mips/mips32/nofpu/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/mips/mips64/n32/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/mips/mips64/n64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc32/fpu/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc32/nofpu/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/s390/s390-32/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/s390/s390-64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/sh/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/sparc/sparc32/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/sparc/sparc64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/tile/tilegx/tilegx32/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/tile/tilegx/tilegx64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/tile/tilepro/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/x86_64/64/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/x86_64/x32/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/hppa/libc.abilist [GLIBC_2.22]: Likewise. * sysdeps/unix/sysv/linux/nios2/libc.abilist [GLIBC_2.22]: Likewise.
204 lines
5.0 KiB
C
204 lines
5.0 KiB
C
/* Copyright (C) 2009-2015 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C 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
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#include <errno.h>
|
|
#include <libintl.h>
|
|
#include <signal.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <not-cancel.h>
|
|
|
|
|
|
#define MF(l) MF1 (l)
|
|
#define MF1(l) str_##l
|
|
#define C(s1, s2) C1 (s1, s2)
|
|
#define C1(s1, s2) s1##s2
|
|
|
|
#define NOW SIGILL
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGFPE
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGSEGV
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGBUS
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGTRAP
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGCLD
|
|
#include "psiginfo-define.h"
|
|
|
|
#define NOW SIGPOLL
|
|
#include "psiginfo-define.h"
|
|
|
|
|
|
/* Print out on stderr a line consisting of the test in S, a colon, a space,
|
|
a message describing the meaning of the signal number PINFO and a newline.
|
|
If S is NULL or "", the colon and space are omitted. */
|
|
void
|
|
psiginfo (const siginfo_t *pinfo, const char *s)
|
|
{
|
|
char buf[512];
|
|
FILE *fp = __fmemopen (buf, sizeof (buf), "w");
|
|
if (fp == NULL)
|
|
{
|
|
const char *colon;
|
|
|
|
if (s == NULL || *s == '\0')
|
|
s = colon = "";
|
|
else
|
|
colon = ": ";
|
|
|
|
__fxprintf (NULL, "%s%ssignal %d\n", s, colon, pinfo->si_signo);
|
|
return;
|
|
}
|
|
|
|
if (s != NULL && *s != '\0')
|
|
fprintf (fp, "%s: ", s);
|
|
|
|
const char *desc;
|
|
if (pinfo->si_signo >= 0 && pinfo->si_signo < NSIG
|
|
&& ((desc = _sys_siglist[pinfo->si_signo]) != NULL
|
|
#ifdef SIGRTMIN
|
|
|| (pinfo->si_signo >= SIGRTMIN && pinfo->si_signo < SIGRTMAX)
|
|
#endif
|
|
))
|
|
{
|
|
#ifdef SIGRTMIN
|
|
if (desc == NULL)
|
|
{
|
|
if (pinfo->si_signo - SIGRTMIN < SIGRTMAX - pinfo->si_signo)
|
|
{
|
|
if (pinfo->si_signo == SIGRTMIN)
|
|
fprintf (fp, "SIGRTMIN (");
|
|
else
|
|
fprintf (fp, "SIGRTMIN+%d (", pinfo->si_signo - SIGRTMIN);
|
|
}
|
|
else
|
|
{
|
|
if (pinfo->si_signo == SIGRTMAX)
|
|
fprintf (fp, "SIGRTMAX (");
|
|
else
|
|
fprintf (fp, "SIGRTMAX-%d (", SIGRTMAX - pinfo->si_signo);
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
fprintf (fp, "%s (", _(desc));
|
|
|
|
const char *base = NULL;
|
|
const uint8_t *offarr = NULL;
|
|
size_t offarr_len = 0;
|
|
switch (pinfo->si_signo)
|
|
{
|
|
#define H(sig) \
|
|
case sig: \
|
|
base = C(codestrs_, sig).str; \
|
|
offarr = C (codes_, sig); \
|
|
offarr_len = sizeof (C (codes_, sig)) / sizeof (C (codes_, sig)[0]);\
|
|
break
|
|
|
|
H (SIGILL);
|
|
H (SIGFPE);
|
|
H (SIGSEGV);
|
|
H (SIGBUS);
|
|
H (SIGTRAP);
|
|
H (SIGCHLD);
|
|
H (SIGPOLL);
|
|
}
|
|
|
|
const char *str = NULL;
|
|
if (offarr != NULL
|
|
&& pinfo->si_code >= 1 && pinfo->si_code <= offarr_len)
|
|
str = base + offarr[pinfo->si_code - 1];
|
|
else
|
|
switch (pinfo->si_code)
|
|
{
|
|
case SI_USER:
|
|
str = N_("Signal sent by kill()");
|
|
break;
|
|
case SI_QUEUE:
|
|
str = N_("Signal sent by sigqueue()");
|
|
break;
|
|
case SI_TIMER:
|
|
str = N_("Signal generated by the expiration of a timer");
|
|
break;
|
|
case SI_ASYNCIO:
|
|
str = N_("\
|
|
Signal generated by the completion of an asynchronous I/O request");
|
|
break;
|
|
case SI_MESGQ:
|
|
str = N_("\
|
|
Signal generated by the arrival of a message on an empty message queue");
|
|
break;
|
|
#ifdef SI_TKILL
|
|
case SI_TKILL:
|
|
str = N_("Signal sent by tkill()");
|
|
break;
|
|
#endif
|
|
#ifdef SI_ASYNCNL
|
|
case SI_ASYNCNL:
|
|
str = N_("\
|
|
Signal generated by the completion of an asynchronous name lookup request");
|
|
break;
|
|
#endif
|
|
#ifdef SI_SIGIO
|
|
case SI_SIGIO:
|
|
str = N_("\
|
|
Signal generated by the completion of an I/O request");
|
|
break;
|
|
#endif
|
|
#ifdef SI_KERNEL
|
|
case SI_KERNEL:
|
|
str = N_("Signal sent by the kernel");
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
if (str != NULL)
|
|
fprintf (fp, "%s ", _(str));
|
|
else
|
|
fprintf (fp, "%d ", pinfo->si_code);
|
|
|
|
if (pinfo->si_signo == SIGILL || pinfo->si_signo == SIGFPE
|
|
|| pinfo->si_signo == SIGSEGV || pinfo->si_signo == SIGBUS)
|
|
fprintf (fp, "[%p])\n", pinfo->si_addr);
|
|
else if (pinfo->si_signo == SIGCHLD)
|
|
fprintf (fp, "%ld %d %ld)\n",
|
|
(long int) pinfo->si_pid, pinfo->si_status,
|
|
(long int) pinfo->si_uid);
|
|
else if (pinfo->si_signo == SIGPOLL)
|
|
fprintf (fp, "%ld)\n", (long int) pinfo->si_band);
|
|
else
|
|
fprintf (fp, "%ld %ld)\n",
|
|
(long int) pinfo->si_pid, (long int) pinfo->si_uid);
|
|
}
|
|
else
|
|
fprintf (fp, _("Unknown signal %d\n"), pinfo->si_signo);
|
|
|
|
fclose (fp);
|
|
|
|
write_not_cancel (STDERR_FILENO, buf, strlen (buf));
|
|
}
|