mes/include/stdarg.h
Andrius Štikonas 542f015135 lib: Fix va_start macro to work for 64-bit arches.
* include/stdarg.h (va_start): Fix to point to correct location.
2023-11-05 09:45:31 +01:00

94 lines
3.6 KiB
C

/* -*-comment-start: "//";comment-end:""-*-
* GNU Mes --- Maxwell Equations of Software
* Copyright © 2017,2018,2019 Jan (janneke) Nieuwenhuizen <janneke@gnu.org>
* Copyright © 2021 W. J. van der Laan <laanwj@protonmail.com>
* Copyright © 2023 Andrius Štikonas <andrius@stikonas.eu>
*
* This file is part of GNU Mes.
*
* GNU Mes 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 of the License, or (at
* your option) any later version.
*
* GNU Mes 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.
*
* You should have received a copy of the GNU General Public License
* along with GNU Mes. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef __MES_STDARG_H
#define __MES_STDARG_H 1
#if SYSTEM_LIBC
#undef __MES_STDARG_H
#include_next <stdarg.h>
#define va_arg8(ap, type) va_arg (ap, type)
#elif (__GNUC__ || __TINYC__) && __riscv
// GCC on RISC-V always passes arguments in registers. Implementing
// these macros without the use of built-ins would be very involved.
// TINYCC tries to be GCC compatible in this case.
#if __TINYC__ < 928
// Bootstrappable TINYCC (version < 928) needs some definitions in RISC-V
typedef char *__builtin_va_list;
#define __va_reg_size (__riscv_xlen >> 3)
#define _tcc_align(addr, type) \
(((unsigned long)addr + __alignof__(type) - 1) \
& -(__alignof__(type)))
#define __builtin_va_arg(ap, type) \
(*(sizeof (type) > (2*__va_reg_size) \
? *(type **)((ap += __va_reg_size) - __va_reg_size) \
: (ap = (va_list)(_tcc_align (ap, type) \
+ (sizeof (type) + __va_reg_size - 1) \
& -__va_reg_size), \
(type *)(ap - ((sizeof (type)+ __va_reg_size - 1) \
& -__va_reg_size)))))
#define __builtin_va_end(ap) (void)(ap)
#if !defined (__builtin_va_copy)
#define __builtin_va_copy(dest, src) (dest) = (src)
#endif
#endif // __TINYC__ < 928
typedef __builtin_va_list va_list;
#define va_start(v, l) __builtin_va_start (v, l)
#define va_end(v) __builtin_va_end (v)
#define va_arg(v, l) __builtin_va_arg (v, l)
#define va_arg8(ap, type) va_arg (ap, type)
#define va_copy(d, s) __builtin_va_copy (d, s)
#else // ! SYSTEM_LIBC && ! __riscv
#include <sys/types.h>
#if __GNUC__ && __x86_64__
#define __FOO_VARARGS 1
#endif
typedef char *va_list;
#define va_start(ap, last) (void)((ap) = (char*)(&(last)) + sizeof (void*))
#define va_arg(ap, type) (type)(((long*)((ap) = ((ap) + sizeof (void*))))[-1])
#define va_align(ap, alignment) ((char*)((((unsigned long) (ap)) + (alignment) - 1) &~ ((alignment) - 1)))
#define va_arg8(ap, type) (type)(((double*)((ap) = (va_align((ap), 8) + sizeof(double))))[-1])
#define va_end(ap) (void)((ap) = 0)
#define va_copy(dest, src) dest = src
int vexec (char const *file_name, va_list ap);
int vfprintf (FILE * stream, char const *template, va_list ap);
int vfscanf (FILE * stream, char const *template, va_list ap);
int vprintf (char const *format, va_list ap);
int vsprintf (char *str, char const *format, va_list ap);
int vsnprintf (char *str, size_t size, char const *format, va_list ap);
int vsscanf (char const *s, char const *template, va_list ap);
#endif // ! SYSTEM_LIBC
#endif // __MES_STDARG_H