mlibc: Tinycc support: gcc -nostdinc -nostdlib.

* mlibc/libc-gcc+tcc.c: New file.
* mlibc/include/dlfcn.h:
* mlibc/include/errno.h:
* mlibc/include/signal.h:
* mlibc/include/sys/mman.h:
* mlibc/include/sys/time.h:  Add tcc declarations.
This commit is contained in:
Jan Nieuwenhuizen 2017-07-27 23:44:22 +02:00
parent ab74876131
commit 397d7a6c43
10 changed files with 450 additions and 35 deletions

View file

@ -62,8 +62,8 @@ exec ${GUILE-guile} --no-auto-compile -L . -L guile -C . -C guile -s "$0" ${1+"$
((eq? libc mini-libc-mes.E) "mini-") ((eq? libc mini-libc-mes.E) "mini-")
(else "")) "guile") #:exit exit))) (else "")) "guile") #:exit exit)))
;;(add-scaffold-test "t" #:libc mini-libc-mes.E) (add-scaffold-test "t" #:libc mini-libc-mes.E)
(add-scaffold-test "t" #:libc libc-mes+tcc.E) ;;(add-scaffold-test "t" #:libc libc-mes+tcc.E)
;; tests/00: exit, functions without libc ;; tests/00: exit, functions without libc
(add-scaffold-test "00-exit-0" #:libc #f) (add-scaffold-test "00-exit-0" #:libc #f)

View file

@ -23,7 +23,22 @@
#if __GNUC__ && POSIX #if __GNUC__ && POSIX
#undef __MES_DLFCN_H #undef __MES_DLFCN_H
#include_next <dlfcn.h> #include_next <dlfcn.h>
#endif // (__GNUC__ && POSIX)
#else // !(__GNUC__ && POSIX)
#define RTLD_LAZY 0x00001
#define RTLD_NOW 0x00002
#define RTLD_BINDING_MASK 0x3
#define RTLD_NOLOAD 0x00004
#define RTLD_DEEPBIND 0x00008
#define RTLD_GLOBAL 0x00100
#define RTLD_LOCAL 0
#define RTLD_NODELETE 0x01000
void *dlopen (char const *filename, int flags);
int dlclose (void *handle);
#endif // !(__GNUC__ && POSIX)
#endif // __MES_DLFCN_H #endif // __MES_DLFCN_H

View file

@ -26,6 +26,8 @@
#endif #endif
#undef __MES_ERRNO_H #undef __MES_ERRNO_H
#include_next <errno.h> #include_next <errno.h>
#else // ! (__GNUC__ && POSIX)
extern int errno;
#endif // ! (__GNUC__ && POSIX) #endif // ! (__GNUC__ && POSIX)
#endif // __MES_ERRNO_H #endif // __MES_ERRNO_H

View file

@ -25,6 +25,184 @@
#include_next <signal.h> #include_next <signal.h>
#else //! (__GNUC__ && POSIX) #else //! (__GNUC__ && POSIX)
typedef int sigset_t; typedef int sigset_t;
typedef int stack_t;
typedef int pid_t;
typedef int uid_t;
typedef int clock_t;
typedef int sigval_t;
#define SIGHUP 1
#define SIGINT 2
#define SIGQUIT 3
#define SIGILL 4
#define SIGTRAP 5
#define SIGABRT 6
#define SIGIOT 6
#define SIGBUS 7
#define SIGFPE 8
#define SIGKILL 9
#define SIGUSR1 10
#define SIGSEGV 11
#define SIGUSR2 12
#define SIGPIPE 13
#define SIGALRM 14
#define SIGTERM 15
#define SIGSTKFLT 16
#define SIGCHLD 17
#define SIGCONT 18
#define SIGSTOP 19
#define SIGTSTP 20
#define SIGTTIN 21
#define SIGTTOU 22
#define SIGURG 23
#define SIGXCPU 24
#define SIGXFSZ 25
#define SIGVTALRM 26
#define SIGPROF 27
#define SIGWINCH 28
#define SIGIO 29
#define SIGPOLL SIGIO
#define FPE_INTDIV 1
#define FPE_INTOVF 2
#define FPE_FLTDIV 3
#define FPE_FLTOVF 4
#define FPE_FLTUND 5
#define FPE_FLTRES 6
#define FPE_FLTINV 7
#define FPE_FLTSUB 8
#define SA_NOCLDSTOP 0x00000001
#define SA_NOCLDWAIT 0x00000002
#define SA_SIGINFO 0x00000004
#define SA_ONSTACK 0x08000000
#define SA_RESTART 0x10000000
#define SA_NODEFER 0x40000000
#define SA_RESETHAND 0x80000000
#define SA_NOMASK SA_NODEFER
#define SA_ONESHOT SA_RESETHAND
typedef struct siginfo_t {
int si_signo;
int si_errno;
int si_code;
int si_trapno;
pid_t si_pid;
uid_t si_uid;
int si_status;
clock_t si_utime;
clock_t si_stime;
sigval_t si_value;
int si_int;
void *si_ptr;
int si_overrun;
int si_timerid;
void *si_addr;
long si_band;
int si_fd;
short si_addr_lsb;
void *si_lower;
void *si_upper;
int si_pkey;
void *si_call_addr;
int si_syscall;
unsigned int si_arch;
} siginfo_t;
// typedef void __signalfn_t(int);
// typedef __signalfn_t *__sighandler_t;
struct sigaction {
void (*sa_sigaction) (int, siginfo_t *, void *);
//__sighandler_t sa_handler;
unsigned long sa_flags;
sigset_t sa_mask;
};
#ifdef __i386__
#define EBX 0
#define ECX 1
#define EDX 2
#define ESI 3
#define EDI 4
#define EBP 5
#define EAX 6
#define DS 7
#define ES 8
#define FS 9
#define GS 10
#define ORIG_EAX 11
#define EIP 12
#define CS 13
#define EFL 14
#define UESP 15
#define SS 16
#define FRAME_SIZE 17
/* Type for general register. */
typedef int greg_t;
/* Number of general registers. */
#define NGREG 19
/* Container for all general registers. */
typedef greg_t gregset_t[NGREG];
/* Definitions taken from the kernel headers. */
struct _libc_fpreg
{
unsigned short int significand[4];
unsigned short int exponent;
};
struct _libc_fpstate
{
unsigned long int cw;
unsigned long int sw;
unsigned long int tag;
unsigned long int ipoff;
unsigned long int cssel;
unsigned long int dataoff;
unsigned long int datasel;
struct _libc_fpreg _st[8];
unsigned long int status;
};
/* Structure to describe FPU registers. */
typedef struct _libc_fpstate *fpregset_t;
typedef struct
{
gregset_t gregs;
/* Due to Linux's history we have to use a pointer here. The SysV/i386
ABI requires a struct with the values. */
fpregset_t fpregs;
unsigned long int oldmask;
unsigned long int cr2;
} mcontext_t;
/* Userlevel context. */
typedef struct ucontext
{
unsigned long int uc_flags;
struct ucontext *uc_link;
stack_t uc_stack;
mcontext_t uc_mcontext;
sigset_t uc_sigmask;
struct _libc_fpstate __fpregs_mem;
} ucontext_t;
#endif // !__i386__
int sigaction (int signum, struct sigaction const *act, struct sigaction *oldact);
int sigemptyset (sigset_t *set);
#endif //! (__GNUC__ && POSIX) #endif //! (__GNUC__ && POSIX)
#endif // __MES_SIGNAL_H #endif // __MES_SIGNAL_H

View file

@ -23,7 +23,21 @@
#if __GNUC__ && POSIX #if __GNUC__ && POSIX
#undef __MES_SYS_MMAN_H #undef __MES_SYS_MMAN_H
#include_next <sys/mman.h> #include_next <sys/mman.h>
#endif // (__GNUC__ && POSIX) #else // !(__GNUC__ && POSIX)
#ifndef __MES_SIZE_T
#define __MES_SIZE_T
typedef unsigned long size_t;
#endif
#define PROT_NONE 0
#define PROT_READ 1
#define PROT_WRITE 2
#define PROT_EXEC 4
int mprotect (void *addr, size_t len, int prot);
#endif // !(__GNUC__ && POSIX)
#endif // __MES_SYS_MMAN_H #endif // __MES_SYS_MMAN_H

View file

@ -23,7 +23,22 @@
#if __GNUC__ && POSIX #if __GNUC__ && POSIX
#undef __MES_SYS_TIME_H #undef __MES_SYS_TIME_H
#include_next <sys/time.h> #include_next <sys/time.h>
#endif // (__GNUC__ && POSIX)
#else // !(__GNUC__ && POSIX)
struct timeval {
long tv_sec;
long tv_usec;
};
struct timezone {
int tz_minuteswest;
int tz_dsttime;
};
int gettimeofday (struct timeval *tv, struct timezone *tz);
#endif // !(__GNUC__ && POSIX)
#endif // __MES_SYS_TIME_H #endif // __MES_SYS_TIME_H

183
mlibc/libc-gcc+tcc.c Normal file
View file

@ -0,0 +1,183 @@
/* -*-comment-start: "//";comment-end:""-*-
* Mes --- Maxwell Equations of Software
* Copyright © 2016,2017 Jan Nieuwenhuizen <janneke@gnu.org>
*
* This file is part of Mes.
*
* 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.
*
* 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 Mes. If not, see <http://www.gnu.org/licenses/>.
*/
#define FULL_MALLOC 1
#include <libc-gcc.c>
#include <libc-mes+tcc.c>
int errno;
#include <unistd.h>
#include <stdarg.h>
#include <signal.h>
#include <sys/mman.h>
#include <sys/time.h>
#define SYS_exit "0x01"
#define SYS_read "0x03"
#define SYS_write "0x04"
#define SYS_open "0x05"
#define SYS_close "0x06"
#define SYS_unlink "0x0a"
#define SYS_lseek "0x13"
#define SYS_access "0x21"
#define SYS_brk "0x2d"
#define SYS_fsync "0x76"
#define SYS_getcwd "0xb7"
int
close (int fd)
{
int r;
asm (
"mov %0,%%ebx\n\t"
"mov $"SYS_close",%%eax\n\t"
"int $0x80"
: "=r" (r)
: "" (fd)
);
return r;
}
int
unlink (char const *file_name)
{
int r;
asm (
"mov %0,%%ebx\n\t"
"mov $"SYS_unlink",%%eax\n\t"
"int $0x80"
: "=r" (r)
: "" (file_name)
);
return r;
}
off_t
lseek (int fd, off_t offset, int whence)
{
int r;
asm (
"mov %1,%%ebx\n\t"
"mov %2,%%ecx\n\t"
"mov %3,%%edx\n\t"
"mov $"SYS_lseek",%%eax\n\t"
"int $0x80\n\t"
"mov %%eax,%0\n\t"
: "=r" (r)
: "" (fd), "" (offset), "" (whence)
: "eax", "ebx", "ecx", "edx"
);
return r;
}
char *
getcwd (char *buf, size_t size)
{
int r;
asm (
"mov %1,%%ebx\n\t"
"mov %2,%%ecx\n\t"
"mov $"SYS_getcwd",%%eax\n\t"
"int $0x80\n\t"
"mov %%eax,%0\n\t"
: "=r" (r)
: "" (buf), "" (size)
: "eax", "ebx", "ecx"
);
return r;
}
int dlclose (void *handle)
{
return 0;
}
void *
dlopen (char const *filename, int flags)
{
return 0;
}
int
mprotect (void *addr, size_t len, int prot)
{
return 0;
}
int
sigaction (int signum, struct sigaction const *act, struct sigaction *oldact)
{
return 0;
}
int
sigemptyset (sigset_t *set)
{
return 0;
}
char *
strcat (char *dest, char const *src)
{
return 0;
}
int
vfprintf (FILE* f, char const* format, va_list ap)
{
int fd = (int)f;
char const *p = format;
while (*p)
if (*p != '%')
putchar (*p++);
else
{
p++;
char c = *p;
switch (c)
{
case '%': {fputc (*p, fd); break;}
case 'c': {char c; c = va_arg (ap, char); fputc (c, fd); break;}
case 'd': {int d; d = va_arg (ap, int); fputs (itoa (d), fd); break;}
case 's': {char *s; s = va_arg (ap, char *); fputs (s, fd); break;}
default: {fputc (*p, fd); break;}
}
p++;
}
va_end (ap);
return 0;
}
int
__udivdi3 (int a, int b)
{
return a / b;
}
int
__umoddi3 (int a, int b)
{
return a % b;
}

View file

@ -181,31 +181,28 @@ putchar (int c)
return 0; return 0;
} }
void *g_malloc_base = 0; char *g_brk = 0;
void * void *
malloc (size_t size) malloc (size_t size)
{ {
void *p = brk (0); if (!g_brk)
if (!g_malloc_base) g_malloc_base = p; g_brk = brk (0);
brk (p+size); if (brk (g_brk + size) == -1)
return 0;
char *p = g_brk;
g_brk += size;
return p; return p;
} }
#if !FULL_MALLOC
void * void *
realloc (void *p, size_t size) realloc (void *p, size_t size)
{ {
(void)p; brk (g_brk + size);
brk (g_malloc_base + size); return g_brk;
return g_malloc_base;
}
void
free (void *p)
{
int *n = (int*)p-1;
//munmap ((void*)p, *n);
} }
#endif
size_t size_t
strlen (char const* s) strlen (char const* s)
@ -504,6 +501,7 @@ fdungetc (int c, int fd)
#endif // POSIX #endif // POSIX
#if __GNUC__ && !POSIX #if __GNUC__ && !POSIX
void void
_start () _start ()
{ {
@ -537,4 +535,5 @@ _start ()
); );
exit (r); exit (r);
} }
#endif // __GNUC__ && !POSIX #endif // __GNUC__ && !POSIX

View file

@ -18,19 +18,21 @@
* along with Mes. If not, see <http://www.gnu.org/licenses/>. * along with Mes. If not, see <http://www.gnu.org/licenses/>.
*/ */
#define FULL_MALLOC 1
#include <libc-mes.c>
#include <setjmp.h> #include <setjmp.h>
#include <stdarg.h> #include <stdarg.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <time.h> #include <time.h>
#include <sys/time.h>
#include <unistd.h> #include <unistd.h>
void #if !__GNUC__
close () #define FULL_MALLOC 1
#include <libc-mes.c>
int
close (int fd)
{ {
asm ("mov____0x8(%ebp),%ebx !8"); asm ("mov____0x8(%ebp),%ebx !8");
@ -38,16 +40,6 @@ close ()
asm ("int____$0x80"); asm ("int____$0x80");
} }
char *
getcwd (char *buf, size_t size)
{
asm ("mov____0x8(%ebp),%ebx !8");
asm ("mov____0x8(%ebp),%ecx !12");
asm ("mov____$i32,%eax SYS_getcwd");
asm ("int____$0x80");
}
int int
unlink (char const *file_name) unlink (char const *file_name)
{ {
@ -68,6 +60,17 @@ lseek (int fd, off_t offset, int whence)
asm ("int____$0x80"); asm ("int____$0x80");
} }
char *
getcwd (char *buf, size_t size)
{
asm ("mov____0x8(%ebp),%ebx !8");
asm ("mov____0x8(%ebp),%ecx !12");
asm ("mov____$i32,%eax SYS_getcwd");
asm ("int____$0x80");
}
#endif // !__GNUC__
int int
execvp (char const *file, char *const argv[]) execvp (char const *file, char *const argv[])
@ -134,6 +137,12 @@ fwrite (void const *ptr, size_t size, size_t nmemb, FILE *stream)
return 0; return 0;
} }
int
gettimeofday (struct timeval *tv, struct timezone *tz)
{
return 0;
}
struct tm * struct tm *
localtime (time_t const *timep) localtime (time_t const *timep)
{ {

View file

@ -322,7 +322,7 @@ malloc (size_t size)
#if !FULL_MALLOC #if !FULL_MALLOC
void * void *
realloc (void *p, int size) realloc (void *p, size_t size)
{ {
brk (g_brk + size); brk (g_brk + size);
return g_brk; return g_brk;