forked from KolibriOS/kolibrios
Delete old kolibri-libc
git-svn-id: svn://kolibrios.org@8696 a494cfbc-eb01-0410-851d-a64ba20cac60
This commit is contained in:
parent
50fc6608b2
commit
99dce7afaf
@ -1,2 +0,0 @@
|
|||||||
-I
|
|
||||||
source/include/
|
|
Binary file not shown.
@ -1,25 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Copyright maxcodehack, 2021
|
|
||||||
# gcc options for build for KolibriOS
|
|
||||||
|
|
||||||
# We start w/o param?
|
|
||||||
if [ -z "$*" ]
|
|
||||||
then
|
|
||||||
echo -e "\033[1;31mfatal error: \033[0mno input files"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# KLIBC variable exists?
|
|
||||||
if [ -z "$KLIBC" ]
|
|
||||||
then
|
|
||||||
echo -e "\033[1;31mfatal error: \033[0mKLIBC variable not set"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# GCC Flags for KolibriOS
|
|
||||||
FLAGS="-c -m32 -nostdinc -fno-common -fno-builtin -fno-leading-underscore -fno-pie -fno-stack-protector -fno-stack-check -mpreferred-stack-boundary=2"
|
|
||||||
KOS_PLATFORM="-Ulinux -U__linux -U__linux__ -U__gnu_linux__ -U__unix__ -U__unix -Uunix -Dkolibri -D__kolibri -D__kolibri__"
|
|
||||||
|
|
||||||
# And, execute gcc:
|
|
||||||
gcc $FLAGS $KOS_PLATFORM -I$KLIBC/source/include $*
|
|
@ -1,24 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Copyright maxcodehack, 2021
|
|
||||||
# ld options for build for KolibriOS
|
|
||||||
|
|
||||||
# We start w/o param?
|
|
||||||
if [ -z "$*" ]
|
|
||||||
then
|
|
||||||
echo "kld: no input files"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# KLIBC variable exists?
|
|
||||||
if [ -z "$KLIBC" ]
|
|
||||||
then
|
|
||||||
echo "kld: KLIBC variable not set"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# LD Flags for KolibriOS
|
|
||||||
FLAGS="-m elf_i386 -nostdlib"
|
|
||||||
|
|
||||||
# And, execute ld:
|
|
||||||
ld $FLAGS -L$KLIBC/lib -T$KLIBC/static.lds $KLIBC/lib/crt0.o $*
|
|
Binary file not shown.
@ -1,50 +0,0 @@
|
|||||||
// MKEXP
|
|
||||||
// Copyright (C) maxcodehack, 2021
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
int main(int argc, char** argv) {
|
|
||||||
if (argc != 3) {
|
|
||||||
printf("usage: %s [symbols.txt exports.c]\n", argv[0]);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
FILE *input, *output;
|
|
||||||
if ((input = fopen(argv[1], "r")) == NULL)
|
|
||||||
{
|
|
||||||
printf("error: file \"%s\" not found\n", argv[1]);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
char buf[10000];
|
|
||||||
|
|
||||||
// Head
|
|
||||||
strcpy(buf, \
|
|
||||||
"#include <math.h>\n" \
|
|
||||||
"#include <stdio.h>\n" \
|
|
||||||
"#include <string.h>\n" \
|
|
||||||
"#include <stdlib.h>\n" \
|
|
||||||
"#include <time.h>\n" \
|
|
||||||
"#include <sys/dirent.h>\n" \
|
|
||||||
"#include <shell_api.h>\n" \
|
|
||||||
"#include <ksys.h>\n\n" \
|
|
||||||
"ksys_coff_etable_t EXPORTS[] = {\n");
|
|
||||||
|
|
||||||
// Generate
|
|
||||||
char symbol[256];
|
|
||||||
while(fscanf(input, "%s", symbol) != EOF) {
|
|
||||||
char temp[256];
|
|
||||||
sprintf(temp, "{\"%s\", %s},\n", symbol, symbol);
|
|
||||||
strcat(buf, temp);
|
|
||||||
}
|
|
||||||
strcat(buf, "NULL,\n};");
|
|
||||||
fclose(input);
|
|
||||||
|
|
||||||
// Output generated
|
|
||||||
output = fopen(argv[2], "w");
|
|
||||||
fputs(buf, output);
|
|
||||||
fclose(output);
|
|
||||||
|
|
||||||
printf("Done, check %s!\n", argv[2]);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
Binary file not shown.
@ -1,46 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
typedef struct List_s {
|
|
||||||
char *this;
|
|
||||||
struct List_s *next;
|
|
||||||
} List;
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
List *root;
|
|
||||||
for (List **pitem = &root;; pitem = &(*pitem)->next) {
|
|
||||||
size_t n = 1024;
|
|
||||||
*pitem = calloc(1, sizeof(List));
|
|
||||||
List *item = *pitem;
|
|
||||||
item->this = calloc(1, n);
|
|
||||||
fgets(item->this, n, stdin);
|
|
||||||
if (item->this[0] == '\n') {
|
|
||||||
free(*pitem);
|
|
||||||
*pitem = NULL;
|
|
||||||
break;
|
|
||||||
} else {
|
|
||||||
item->this[strlen(item->this) - 1] = '\0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (List *item = root; item; item = item->next) {
|
|
||||||
char asm_name[255];
|
|
||||||
sprintf(asm_name, "%s.asm", item->this);
|
|
||||||
FILE *out = fopen(asm_name, "wb");
|
|
||||||
|
|
||||||
fprintf(out, "format ELF\n");
|
|
||||||
fprintf(out, "include \"__lib__.inc\"\n");
|
|
||||||
fprintf(out, "fun equ __func@%s\n", item->this);
|
|
||||||
fprintf(out, "fun_str equ '%s'\n", item->this);
|
|
||||||
fprintf(out, "section '.text'\n");
|
|
||||||
fprintf(out, "fun_name db fun_str, 0\n");
|
|
||||||
fprintf(out, "section '.data'\n");
|
|
||||||
fprintf(out, "extrn lib_name\n");
|
|
||||||
fprintf(out, "public fun as fun_str\n");
|
|
||||||
fprintf(out, "fun dd fun_name\n");
|
|
||||||
fprintf(out, "lib dd lib_name\n");
|
|
||||||
|
|
||||||
fclose(out);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,9 +0,0 @@
|
|||||||
format ELF
|
|
||||||
|
|
||||||
include '__lib__.inc'
|
|
||||||
|
|
||||||
section '.text'
|
|
||||||
|
|
||||||
public lib_name
|
|
||||||
|
|
||||||
lib_name db 0x55, 0xAA, lib_name_str, 0
|
|
@ -1,2 +0,0 @@
|
|||||||
lib_name equ @libc.obj
|
|
||||||
lib_name_str equ '/sys/lib/libc.obj'
|
|
@ -1,24 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -e
|
|
||||||
|
|
||||||
MKLIB=../linuxtools/mklib
|
|
||||||
AR=ar
|
|
||||||
FASM=fasm
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
echo "Generate ASM files..."
|
|
||||||
cat symbols.txt | $MKLIB
|
|
||||||
|
|
||||||
echo "Compile ASM files..."
|
|
||||||
for asm_file in $(find *.asm)
|
|
||||||
do
|
|
||||||
$FASM $asm_file > /dev/null
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "Create libc.obj.a library..."
|
|
||||||
ar -rsc ../lib/libc.obj.a *.o
|
|
||||||
mv __lib__.asm __lib__.asm.bak
|
|
||||||
rm *.o *.asm
|
|
||||||
mv __lib__.asm.bak __lib__.asm
|
|
||||||
echo "Done!"
|
|
@ -1,112 +0,0 @@
|
|||||||
abs
|
|
||||||
acos
|
|
||||||
acosh
|
|
||||||
asin
|
|
||||||
asinh
|
|
||||||
atan
|
|
||||||
atan2
|
|
||||||
atanh
|
|
||||||
atoi
|
|
||||||
atol
|
|
||||||
atoll
|
|
||||||
calloc
|
|
||||||
ceil
|
|
||||||
clearerr
|
|
||||||
closedir
|
|
||||||
cos
|
|
||||||
cosh
|
|
||||||
debug_printf
|
|
||||||
difftime
|
|
||||||
div
|
|
||||||
exit
|
|
||||||
exp
|
|
||||||
fabs
|
|
||||||
fclose
|
|
||||||
feof
|
|
||||||
ferror
|
|
||||||
fflush
|
|
||||||
fgetc
|
|
||||||
fgetpos
|
|
||||||
fgets
|
|
||||||
floor
|
|
||||||
fmod
|
|
||||||
fopen
|
|
||||||
fputc
|
|
||||||
fputs
|
|
||||||
fread
|
|
||||||
free
|
|
||||||
freopen
|
|
||||||
frexp
|
|
||||||
fseek
|
|
||||||
fsetpos
|
|
||||||
ftell
|
|
||||||
fwrite
|
|
||||||
getchar
|
|
||||||
gets
|
|
||||||
hypot
|
|
||||||
itoa
|
|
||||||
labs
|
|
||||||
ldexp
|
|
||||||
ldiv
|
|
||||||
llabs
|
|
||||||
lldiv
|
|
||||||
localtime
|
|
||||||
log
|
|
||||||
malloc
|
|
||||||
memccpy
|
|
||||||
memchr
|
|
||||||
memcmp
|
|
||||||
memcpy
|
|
||||||
memmove
|
|
||||||
memset
|
|
||||||
mktime
|
|
||||||
modf
|
|
||||||
modfl
|
|
||||||
opendir
|
|
||||||
perror
|
|
||||||
pow
|
|
||||||
pow10
|
|
||||||
pow2
|
|
||||||
printf
|
|
||||||
puts
|
|
||||||
readdir
|
|
||||||
realloc
|
|
||||||
remove
|
|
||||||
rename
|
|
||||||
rewind
|
|
||||||
rewinddir
|
|
||||||
seekdir
|
|
||||||
setbuf
|
|
||||||
setvbuf
|
|
||||||
sin
|
|
||||||
sinh
|
|
||||||
snprintf
|
|
||||||
sprintf
|
|
||||||
sqrt
|
|
||||||
strcat
|
|
||||||
strchr
|
|
||||||
strcmp
|
|
||||||
strcoll
|
|
||||||
strcpy
|
|
||||||
strcspn
|
|
||||||
strerror
|
|
||||||
strlen
|
|
||||||
strncat
|
|
||||||
strncmp
|
|
||||||
strncpy
|
|
||||||
strpbrk
|
|
||||||
strrchr
|
|
||||||
strrev
|
|
||||||
strspn
|
|
||||||
strstr
|
|
||||||
strtok
|
|
||||||
strxfrm
|
|
||||||
tan
|
|
||||||
tanh
|
|
||||||
telldir
|
|
||||||
time
|
|
||||||
tmpfile
|
|
||||||
tmpnam
|
|
||||||
vprintf
|
|
||||||
vsnprintf
|
|
||||||
|
|
@ -1,27 +0,0 @@
|
|||||||
KGCC = ../linuxtools/kgcc
|
|
||||||
KLD = ../linuxtools/kld
|
|
||||||
KPACK = kpack
|
|
||||||
KTCC=../../../programs/develop/ktcc/trunk/bin/kos32-tcc
|
|
||||||
KTCC_FLAGS = -nostdlib -stack=1048576 -I$(KLIBC)/source/include -L$(KLIBC)/lib $(KLIBC)/lib/crt0.o
|
|
||||||
|
|
||||||
ifndef NAME
|
|
||||||
NAME = basic_gui
|
|
||||||
endif
|
|
||||||
|
|
||||||
static_kgcc:
|
|
||||||
$(KGCC) $(NAME).c
|
|
||||||
$(KLD) $(NAME).o -o $(NAME) -lc
|
|
||||||
$(KPACK) $(NAME)
|
|
||||||
|
|
||||||
static_tcc:
|
|
||||||
$(KTCC) $(KTCC_FLAGS) $(NAME).c -o $(NAME) -lc
|
|
||||||
$(KPACK) $(NAME)
|
|
||||||
|
|
||||||
dynamic_kgcc:
|
|
||||||
$(KGCC) -D_DYNAMIC $(NAME).c
|
|
||||||
$(KLD) $(NAME).o -o $(NAME) -lc.obj
|
|
||||||
$(KPACK) $(NAME)
|
|
||||||
|
|
||||||
dynamic_tcc:
|
|
||||||
$(KTCC) $(KTCC_FLAGS) -D_DYNAMIC $(NAME).c -o $(NAME)_dyn -lc.obj
|
|
||||||
$(KPACK) $(NAME)_dyn
|
|
@ -1,116 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include "ksys.h"
|
|
||||||
|
|
||||||
ksys_colors_table_t sys_color_table;
|
|
||||||
|
|
||||||
char statusbar[255];
|
|
||||||
ksys_proc_table_t proc_info;
|
|
||||||
char text_line[255];
|
|
||||||
|
|
||||||
enum BUTTONS
|
|
||||||
{
|
|
||||||
BTN_QUIT = 1,
|
|
||||||
BTN_POP = 10,
|
|
||||||
BTN_UNLOCK = 11
|
|
||||||
};
|
|
||||||
|
|
||||||
#define FONT_W 8
|
|
||||||
#define FONT_H 14
|
|
||||||
#define LINES 10
|
|
||||||
|
|
||||||
void draw_window()
|
|
||||||
{
|
|
||||||
int win_hight, win_width, i, pos_y = _ksys_get_skin_height() + 36; // 60 == 24+36
|
|
||||||
|
|
||||||
// start redraw
|
|
||||||
_ksys_start_draw();
|
|
||||||
// define&draw window
|
|
||||||
_ksys_create_window(10, 40, 600, 400, "My window", sys_color_table.work_area, 0x13);
|
|
||||||
_ksys_process_info(&proc_info, -1);
|
|
||||||
|
|
||||||
win_width = proc_info.winx_size;
|
|
||||||
win_hight = proc_info.winy_size;
|
|
||||||
|
|
||||||
_ksys_define_button(10, 30, 70, 20, BTN_POP, sys_color_table.work_button);
|
|
||||||
_ksys_draw_text("BUTTON1", 15, 34, 0, 0x90000000 | sys_color_table.work_button_text); //0x80000000 asciiz
|
|
||||||
|
|
||||||
_ksys_define_button(100, 30, 80, 20, BTN_UNLOCK, sys_color_table.work_button);
|
|
||||||
_ksys_draw_text("BUTTTON2", 110, 34, 0, 0x90000000 | sys_color_table.work_button_text);
|
|
||||||
|
|
||||||
// display statusbar
|
|
||||||
_ksys_draw_bar(6, win_hight - 17, win_width - 11, 12, 0x80000000 | sys_color_table.work_area); //0x80000000 gradient
|
|
||||||
_ksys_draw_text(statusbar, 10, win_hight - 15, 0, 0x80000000 | sys_color_table.work_text);
|
|
||||||
|
|
||||||
// display strings
|
|
||||||
for (i = LINES; i > 0; i--){
|
|
||||||
snprintf (text_line, sizeof text_line, "Line[%d]<<Just a text>>", i);
|
|
||||||
|
|
||||||
text_line[(win_width - 10 - 5) / FONT_W + 1] = '\0'; // clip text size, seems to big lines crashing OS, and form len by window size
|
|
||||||
_ksys_draw_text(text_line, 5, pos_y, 0, 0x90000000 | sys_color_table.work_text);
|
|
||||||
pos_y += FONT_H;
|
|
||||||
|
|
||||||
if(pos_y + 29 > win_hight) break; // 12 font + 12 statusbar + 5 border
|
|
||||||
}
|
|
||||||
// end redraw
|
|
||||||
_ksys_end_draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
int main()
|
|
||||||
{
|
|
||||||
int gui_event;
|
|
||||||
uint32_t pressed_button = 0, mouse_button;
|
|
||||||
ksys_pos_t mouse_pos;
|
|
||||||
strcpy(statusbar, "Program running...Double click on TEXT for details");
|
|
||||||
|
|
||||||
_ksys_get_colors(&sys_color_table);
|
|
||||||
_ksys_set_event_mask(0xC0000027); // mouse events only when focused window and mouse inside
|
|
||||||
|
|
||||||
do{
|
|
||||||
gui_event = _ksys_get_event();
|
|
||||||
switch(gui_event)
|
|
||||||
{
|
|
||||||
case KSYS_EVENT_NONE:
|
|
||||||
break;
|
|
||||||
case KSYS_EVENT_REDRAW:
|
|
||||||
draw_window();
|
|
||||||
break;
|
|
||||||
case KSYS_EVENT_KEY:
|
|
||||||
break;
|
|
||||||
case KSYS_EVENT_BUTTON:
|
|
||||||
pressed_button = _ksys_get_button();
|
|
||||||
switch (pressed_button)
|
|
||||||
{
|
|
||||||
case BTN_POP:
|
|
||||||
strcpy(statusbar, "POP pressed....");
|
|
||||||
draw_window();
|
|
||||||
break;
|
|
||||||
case BTN_UNLOCK:
|
|
||||||
strcpy(statusbar, "UNLOCK pressed....");
|
|
||||||
draw_window();
|
|
||||||
break;
|
|
||||||
case BTN_QUIT:
|
|
||||||
return 0;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case KSYS_EVENT_MOUSE:
|
|
||||||
mouse_pos = _ksys_get_mouse_pos(KSYS_MOUSE_WINDOW_POS); // window relative
|
|
||||||
mouse_button = _ksys_get_mouse_eventstate();
|
|
||||||
debug_printf("mouse ev (%d,%d)%x\n", mouse_pos.x, mouse_pos.y, mouse_button);
|
|
||||||
if (mouse_button & (1<<24)) // double click
|
|
||||||
{
|
|
||||||
int n = (mouse_pos.y - 60) / FONT_H;
|
|
||||||
if (n < 0 || n >= LINES) break;
|
|
||||||
debug_printf("click on str(%d), clip slot(%d)\n", n, LINES - n - 1);
|
|
||||||
sprintf(statusbar, "click on str(%d), clip slot(%d)\n", n, LINES - n - 1);
|
|
||||||
draw_window();
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} while(1) ; /* End of main activity loop */
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,56 +0,0 @@
|
|||||||
AR = ar -rcs
|
|
||||||
CLINK = ../linuxtools/clink
|
|
||||||
MKEXP = ../linuxtools/mkexp
|
|
||||||
KPACK = kpack
|
|
||||||
|
|
||||||
INCLUDE = include
|
|
||||||
LIB_DIR = ../lib
|
|
||||||
|
|
||||||
LIBC.A = $(LIB_DIR)/libc.a
|
|
||||||
LIBC.OBJ = $(LIB_DIR)/libc.obj
|
|
||||||
|
|
||||||
CFLAGS = -I$(INCLUDE) -m32 -nostdinc -nostdlib -DGNUC -Os -fno-common -fno-builtin -fno-leading-underscore -fno-pie
|
|
||||||
DIRS := string stdlib stdio sys math ctype exports setjmp shell_api
|
|
||||||
|
|
||||||
cfiles := $(foreach dir,$(DIRS),$(patsubst %.c, %.o, $(wildcard $(dir)/*.c)))
|
|
||||||
asmfiles := $(foreach dir,$(DIRS),$(patsubst %.s, %.o, $(wildcard $(dir)/*.s)))
|
|
||||||
.PHONY: clean all
|
|
||||||
|
|
||||||
ifdef windir
|
|
||||||
cfiles := $(subst /,\,$(cfiles))
|
|
||||||
asmfiles := $(subst /,\,$(asmfiles))
|
|
||||||
LIB_DIR := $(subst /,\,$(LIB_DIR))
|
|
||||||
LIBNAME := $(subst /,\,$(LIBNAME))
|
|
||||||
RM = del /F /Q
|
|
||||||
MKDIR_P = md
|
|
||||||
else
|
|
||||||
RM = rm -rf
|
|
||||||
MKDIR_P = mkdir -p
|
|
||||||
endif
|
|
||||||
|
|
||||||
all: $(cfiles) $(asmfiles) $(LIB_DIR) $(LIBC.A)
|
|
||||||
fasm crt/crt0.asm $(LIB_DIR)/crt0.o
|
|
||||||
|
|
||||||
$(LIBC.A): $(cfiles) $(asmfiles)
|
|
||||||
$(AR) $(LIBC.A) $^
|
|
||||||
|
|
||||||
$(LIB_DIR):
|
|
||||||
$(MKDIR_P) $(LIB_DIR)
|
|
||||||
|
|
||||||
$(asmfiles):
|
|
||||||
$(CC) -E -x c -nostdinc -Imath $*.s > $*.sx
|
|
||||||
$(AS) --32 $*.sx -o $*.o
|
|
||||||
$(RM) $*.sx
|
|
||||||
|
|
||||||
clean:
|
|
||||||
$(RM) $(cfiles) $(asmfiles) exports/exports.c
|
|
||||||
$(RM) $(LIBNAME)
|
|
||||||
|
|
||||||
shared: $(cfiles) $(asmfiles)
|
|
||||||
$(CLINK) $^
|
|
||||||
$(KPACK) a.out.obj
|
|
||||||
mv a.out.obj $(LIBC.OBJ)
|
|
||||||
|
|
||||||
test:
|
|
||||||
make -C ../test -f Makefile.tcc
|
|
||||||
kex ../test/test
|
|
@ -1,5 +0,0 @@
|
|||||||
KTCC=../../../programs/develop/ktcc/trunk/bin/kos32-tcc
|
|
||||||
|
|
||||||
all:
|
|
||||||
make clean
|
|
||||||
env CC=$(KTCC) make
|
|
@ -1,9 +0,0 @@
|
|||||||
MKEXP=../linuxtools/mkexp
|
|
||||||
|
|
||||||
all:
|
|
||||||
$(MAKE) clean
|
|
||||||
$(MKEXP) ../loader/symbols.txt exports/exports.c
|
|
||||||
env CC=i686-w64-mingw32-gcc CPP=i686-w64-mingw32-cpp AS=i686-w64-mingw32-as $(MAKE) shared
|
|
||||||
|
|
||||||
install:
|
|
||||||
cp ../lib/libc.obj ~/.kex/root/RD/1/LIB
|
|
@ -1,220 +0,0 @@
|
|||||||
format ELF
|
|
||||||
section '.text' executable
|
|
||||||
public start
|
|
||||||
public start as '_start'
|
|
||||||
;extrn mf_init
|
|
||||||
extrn main
|
|
||||||
;include 'debug2.inc'
|
|
||||||
include '../../../../programs/proc32.inc'
|
|
||||||
include '../../../../programs/macros.inc'
|
|
||||||
include '../../../../programs/dll.inc'
|
|
||||||
__DEBUG__=0
|
|
||||||
|
|
||||||
;start_:
|
|
||||||
virtual at 0
|
|
||||||
db 'MENUET01' ; 1. Magic number (8 bytes)
|
|
||||||
dd 0x01 ; 2. Version of executable file
|
|
||||||
dd start ; 3. Start address
|
|
||||||
imgsz dd 0x0 ; 4. Size of image
|
|
||||||
dd 0x100000 ; 5. Size of needed memory
|
|
||||||
dd 0x100000 ; 6. Pointer to stack
|
|
||||||
hparams dd 0x0 ; 7. Pointer to program arguments
|
|
||||||
hpath dd 0x0 ; 8. Pointer to program path
|
|
||||||
end virtual
|
|
||||||
|
|
||||||
start:
|
|
||||||
;DEBUGF 'Start programm\n'
|
|
||||||
;init heap of memory
|
|
||||||
mov eax,68
|
|
||||||
mov ebx,11
|
|
||||||
int 0x40
|
|
||||||
|
|
||||||
mov [argc], 0
|
|
||||||
mov eax, [hparams]
|
|
||||||
test eax, eax
|
|
||||||
jz .without_path
|
|
||||||
mov eax, path
|
|
||||||
cmp word ptr eax, 32fh ; '/#3' UTF8
|
|
||||||
jne .without_path
|
|
||||||
mov word ptr eax, 12fh ; '/#1' fix to CP866
|
|
||||||
.without_path:
|
|
||||||
mov esi, eax
|
|
||||||
call push_param
|
|
||||||
; retrieving parameters
|
|
||||||
mov esi, params
|
|
||||||
xor edx, edx ; dl - èä¸ò ïàðàìåòð(1) èëè ðàçäåëèòåëè(0)
|
|
||||||
; dh - ñèìâîë ñ êîòîðîãî íà÷àëñÿ ïàðàìåòð (1 êàâû÷êè, 0 îñòàëüíîå)
|
|
||||||
mov ecx, 1 ; cl = 1
|
|
||||||
; ch = 0 ïðîñòî íîëü
|
|
||||||
.parse:
|
|
||||||
lodsb
|
|
||||||
test al, al
|
|
||||||
jz .run
|
|
||||||
test dl, dl
|
|
||||||
jnz .findendparam
|
|
||||||
;{åñëè áûë ðàçäåëèòåëü
|
|
||||||
cmp al, ' '
|
|
||||||
jz .parse ;çàãðóæåí ïðîáåë, ãðóçèì ñëåäóþùèé ñèìâîë
|
|
||||||
mov dl, cl ;íà÷èíàåòñÿ ïàðàìåòð
|
|
||||||
cmp al, '"'
|
|
||||||
jz @f ;çàãðóæåíû êàâû÷êè
|
|
||||||
mov dh, ch ;ïàðàìåòð áåç êàâû÷åê
|
|
||||||
dec esi
|
|
||||||
call push_param
|
|
||||||
inc esi
|
|
||||||
jmp .parse
|
|
||||||
|
|
||||||
@@:
|
|
||||||
mov dh, cl ;ïàðàìåòð â êàâû÷åêàõ
|
|
||||||
call push_param ;åñëè íå ïðîáåë çíà÷èò íà÷èíàåòñÿ êàêîé òî ïàðàìåòð
|
|
||||||
jmp .parse ;åñëè áûë ðàçäåëèòåëü}
|
|
||||||
|
|
||||||
.findendparam:
|
|
||||||
test dh, dh
|
|
||||||
jz @f ; áåç êàâû÷åê
|
|
||||||
cmp al, '"'
|
|
||||||
jz .clear
|
|
||||||
jmp .parse
|
|
||||||
@@:
|
|
||||||
cmp al, ' '
|
|
||||||
jnz .parse
|
|
||||||
|
|
||||||
.clear:
|
|
||||||
lea ebx, [esi - 1]
|
|
||||||
mov [ebx], ch
|
|
||||||
mov dl, ch
|
|
||||||
jmp .parse
|
|
||||||
|
|
||||||
.run:
|
|
||||||
call load_imports
|
|
||||||
push argv
|
|
||||||
push [argc]
|
|
||||||
call main
|
|
||||||
.exit:
|
|
||||||
xor eax,eax
|
|
||||||
dec eax
|
|
||||||
int 0x40
|
|
||||||
dd -1
|
|
||||||
.crash:
|
|
||||||
jmp .exit
|
|
||||||
;============================
|
|
||||||
push_param:
|
|
||||||
;============================
|
|
||||||
;parameters
|
|
||||||
; esi - pointer
|
|
||||||
;description
|
|
||||||
; procedure increase argc
|
|
||||||
; and add pointer to array argv
|
|
||||||
; procedure changes ebx
|
|
||||||
mov ebx, [argc]
|
|
||||||
cmp ebx, max_parameters
|
|
||||||
jae .dont_add
|
|
||||||
mov [argv+4*ebx], esi
|
|
||||||
inc [argc]
|
|
||||||
.dont_add:
|
|
||||||
ret
|
|
||||||
;==============================
|
|
||||||
|
|
||||||
;==============================
|
|
||||||
load_imports:
|
|
||||||
;==============================
|
|
||||||
;parameters
|
|
||||||
; none
|
|
||||||
;description
|
|
||||||
; imports must be located at end of image (but before BSS sections)
|
|
||||||
; the address of end of imports (next byte after imports) is located in imgsz
|
|
||||||
; look at each import from that address up to illegal import
|
|
||||||
; legal import is such that:
|
|
||||||
; first pointer points to procedure name
|
|
||||||
; and is smaller than imgsz
|
|
||||||
; second pointer points lo library name, starting with 0x55, 0xAA
|
|
||||||
; and is smaller than imgsz
|
|
||||||
; each library should be initialized as appropriate, once
|
|
||||||
; so as library is initialized, its name will be replaced 0x00
|
|
||||||
mov ebx, [imgsz] ; byte after imports
|
|
||||||
.handle_next_import:
|
|
||||||
sub ebx, 4 ; ebx = pointer to pointer to library name
|
|
||||||
mov esi, dword[ebx] ; esi = pointer to library name
|
|
||||||
push ebx
|
|
||||||
push esi
|
|
||||||
call load_library ; eax = pointer to library exports
|
|
||||||
pop esi
|
|
||||||
pop ebx
|
|
||||||
test eax, eax
|
|
||||||
jz .done
|
|
||||||
sub ebx, 4 ; ebx = pointer to pointer to symbol name
|
|
||||||
push ebx
|
|
||||||
stdcall dll.GetProcAddress, eax, dword[ebx]
|
|
||||||
pop ebx
|
|
||||||
test eax, eax
|
|
||||||
jz .fail
|
|
||||||
mov dword[ebx], eax
|
|
||||||
jmp .handle_next_import
|
|
||||||
.done:
|
|
||||||
ret
|
|
||||||
.fail:
|
|
||||||
ret
|
|
||||||
;==============================
|
|
||||||
|
|
||||||
;==============================
|
|
||||||
load_library:
|
|
||||||
;==============================
|
|
||||||
;parameters
|
|
||||||
; ebx: library name address
|
|
||||||
;description
|
|
||||||
; each library should be initialized as appropriate, once
|
|
||||||
; so as library is initialized, its name will be replaced 0x00
|
|
||||||
; and 4 next bytes will be set to address of library
|
|
||||||
; first two bytes of library name must be 0x55, 0xAA (is like a magic)
|
|
||||||
cld ; move esi further, not back
|
|
||||||
cmp esi, [imgsz]
|
|
||||||
ja .fail
|
|
||||||
lodsb ; al = first byte of library name
|
|
||||||
cmp al, 0x55
|
|
||||||
jne .fail
|
|
||||||
lodsb ; al = second byte of library name
|
|
||||||
cmp al, 0xAA
|
|
||||||
jne .fail
|
|
||||||
lodsb ; al = third byte of library name (0x00 if the library is already loaded)
|
|
||||||
test al, al
|
|
||||||
jnz .load
|
|
||||||
lodsd ; if we here, then third byte is 0x00 => address of library is in next 4 bytes
|
|
||||||
; now eax contains address of library
|
|
||||||
ret
|
|
||||||
.load:
|
|
||||||
dec esi ; we checked on 0 before, let's go back
|
|
||||||
mov eax, 68
|
|
||||||
mov ebx, 19
|
|
||||||
mov ecx, esi
|
|
||||||
int 0x40 ; eax = address of exports
|
|
||||||
mov byte[esi], 0 ; library is loaded, let's place 0 in first byte of name
|
|
||||||
mov [esi + 1], eax ; now next 4 bytes of library name are replaced by address of library
|
|
||||||
; call lib_init
|
|
||||||
stdcall dll.GetProcAddress, eax, lib_init_str ; eax = address of lib_init
|
|
||||||
test eax, eax
|
|
||||||
jz .ret
|
|
||||||
stdcall dll.Init, eax
|
|
||||||
.ret:
|
|
||||||
mov eax, [esi + 1] ; put address of library into eax
|
|
||||||
ret
|
|
||||||
.fail:
|
|
||||||
mov eax, 0
|
|
||||||
ret
|
|
||||||
;==============================
|
|
||||||
|
|
||||||
lib_init_str db 'lib_init', 0
|
|
||||||
|
|
||||||
public argc as '__argc'
|
|
||||||
public params as '__argv'
|
|
||||||
public path as '__path'
|
|
||||||
|
|
||||||
section '.bss'
|
|
||||||
buf_len = 0x400
|
|
||||||
max_parameters=0x20
|
|
||||||
argc rd 1
|
|
||||||
argv rd max_parameters
|
|
||||||
path rb buf_len
|
|
||||||
params rb buf_len
|
|
||||||
|
|
||||||
;section '.data'
|
|
||||||
;include_debug_strings ; ALWAYS present in data section
|
|
@ -1,21 +0,0 @@
|
|||||||
#include <ctype.h>
|
|
||||||
|
|
||||||
unsigned short __is[129] = {
|
|
||||||
0, /* EOF */
|
|
||||||
0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004,
|
|
||||||
0x004, 0x104, 0x104, 0x104, 0x104, 0x104, 0x004, 0x004,
|
|
||||||
0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004,
|
|
||||||
0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004,
|
|
||||||
0x140, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0,
|
|
||||||
0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0,
|
|
||||||
0x459, 0x459, 0x459, 0x459, 0x459, 0x459, 0x459, 0x459,
|
|
||||||
0x459, 0x459, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0,
|
|
||||||
0x0D0, 0x653, 0x653, 0x653, 0x653, 0x653, 0x653, 0x253,
|
|
||||||
0x253, 0x253, 0x253, 0x253, 0x253, 0x253, 0x253, 0x253,
|
|
||||||
0x253, 0x253, 0x253, 0x253, 0x253, 0x253, 0x253, 0x253,
|
|
||||||
0x253, 0x253, 0x253, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x0D0,
|
|
||||||
0x0D0, 0x473, 0x473, 0x473, 0x473, 0x473, 0x473, 0x073,
|
|
||||||
0x073, 0x073, 0x073, 0x073, 0x073, 0x073, 0x073, 0x073,
|
|
||||||
0x073, 0x073, 0x073, 0x073, 0x073, 0x073, 0x073, 0x073,
|
|
||||||
0x073, 0x073, 0x073, 0x0D0, 0x0D0, 0x0D0, 0x0D0, 0x004
|
|
||||||
};
|
|
@ -1,41 +0,0 @@
|
|||||||
#ifndef _CTYPE_H_
|
|
||||||
#define _CTYPE_H_
|
|
||||||
/*
|
|
||||||
** All character classification functions except isascii().
|
|
||||||
** Integer argument (c) must be in ASCII range (0-127) for
|
|
||||||
** dependable answers.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define __ALNUM 1
|
|
||||||
#define __ALPHA 2
|
|
||||||
#define __CNTRL 4
|
|
||||||
#define __DIGIT 8
|
|
||||||
#define __GRAPH 16
|
|
||||||
#define __LOWER 32
|
|
||||||
#define __PRINT 64
|
|
||||||
#define __PUNCT 128
|
|
||||||
#define __BLANK 256
|
|
||||||
#define __UPPER 512
|
|
||||||
#define __XDIGIT 1024
|
|
||||||
|
|
||||||
extern unsigned short __is[129];
|
|
||||||
|
|
||||||
#define isalnum(c)(__is[c+1] & __ALNUM ) /* 'a'-'z', 'A'-'Z', '0'-'9' */
|
|
||||||
#define isalpha(c)(__is[c+1] & __ALPHA ) /* 'a'-'z', 'A'-'Z' */
|
|
||||||
#define iscntrl(c)(__is[c+1] & __CNTRL ) /* 0-31, 127 */
|
|
||||||
#define isdigit(c)(__is[c+1] & __DIGIT ) /* '0'-'9' */
|
|
||||||
#define isgraph(c)(__is[c+1] & __GRAPH ) /* '!'-'~' */
|
|
||||||
#define islower(c)(__is[c+1] & __LOWER ) /* 'a'-'z' */
|
|
||||||
#define isprint(c)(__is[c+1] & __PRINT ) /* ' '-'~' */
|
|
||||||
#define ispunct(c)(__is[c+1] & __PUNCT ) /* !alnum && !cntrl && !space */
|
|
||||||
#define isspace(c)(__is[c+1] & __BLANK ) /* HT, LF, VT, FF, CR, ' ' */
|
|
||||||
#define isupper(c)(__is[c+1] & __UPPER ) /* 'A'-'Z' */
|
|
||||||
#define isxdigit(c)(__is[c+1] & __XDIGIT) /* '0'-'9', 'a'-'f', 'A'-'F' */
|
|
||||||
|
|
||||||
#define isascii(c) (!((c)&(~0x7f)))
|
|
||||||
#define toascii(c) ((c)&0x7f)
|
|
||||||
|
|
||||||
extern unsigned char tolower(unsigned char c);
|
|
||||||
extern unsigned char toupper(unsigned char c);
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,144 +0,0 @@
|
|||||||
#ifndef _ERRNO_H_
|
|
||||||
#define _ERRNO_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern int _GLOBAL_VAR(errno);
|
|
||||||
|
|
||||||
#define EPERM 1 /* Operation not permitted */
|
|
||||||
#define ENOENT 2 /* No such file or directory */
|
|
||||||
#define ESRCH 3 /* No such process */
|
|
||||||
#define EINTR 4 /* Interrupted system call */
|
|
||||||
#define EIO 5 /* Input/output error */
|
|
||||||
#define ENXIO 6 /* Device not configured */
|
|
||||||
#define E2BIG 7 /* Argument list too long */
|
|
||||||
#define ENOEXEC 8 /* Exec format error */
|
|
||||||
#define EBADF 9 /* Bad file descriptor */
|
|
||||||
#define ECHILD 10 /* No child processes */
|
|
||||||
#define EDEADLK 11 /* Resource deadlock avoided */
|
|
||||||
/* 11 was EAGAIN */
|
|
||||||
#define ENOMEM 12 /* Cannot allocate memory */
|
|
||||||
#define EACCES 13 /* Permission denied */
|
|
||||||
#define EFAULT 14 /* Bad address */
|
|
||||||
#define ENOTBLK 15 /* Block device required */
|
|
||||||
#define EBUSY 16 /* Device / Resource busy */
|
|
||||||
#define EEXIST 17 /* File exists */
|
|
||||||
#define EXDEV 18 /* Cross-device link */
|
|
||||||
#define ENODEV 19 /* Operation not supported by device */
|
|
||||||
#define ENOTDIR 20 /* Not a directory */
|
|
||||||
#define EISDIR 21 /* Is a directory */
|
|
||||||
#define EINVAL 22 /* Invalid argument */
|
|
||||||
#define ENFILE 23 /* Too many open files in system */
|
|
||||||
#define EMFILE 24 /* Too many open files */
|
|
||||||
#define ENOTTY 25 /* Inappropriate ioctl for device */
|
|
||||||
#define ETXTBSY 26 /* Text file busy */
|
|
||||||
#define EFBIG 27 /* File too large */
|
|
||||||
#define ENOSPC 28 /* No space left on device */
|
|
||||||
#define ESPIPE 29 /* Illegal seek */
|
|
||||||
#define EROFS 30 /* Read-only file system */
|
|
||||||
#define EMLINK 31 /* Too many links */
|
|
||||||
#define EPIPE 32 /* Broken pipe */
|
|
||||||
|
|
||||||
/* math software */
|
|
||||||
#define EDOM 33 /* Numerical argument out of domain */
|
|
||||||
#define ERANGE 34 /* Result too large */
|
|
||||||
|
|
||||||
/* non-blocking and interrupt i/o */
|
|
||||||
#define EAGAIN 35 /* Resource temporarily unavailable */
|
|
||||||
#define EWOULDBLOCK EAGAIN /* Operation would block */
|
|
||||||
#define EINPROGRESS 36 /* Operation now in progress */
|
|
||||||
#define EALREADY 37 /* Operation already in progress */
|
|
||||||
|
|
||||||
/* ipc/network software -- argument errors */
|
|
||||||
#define ENOTSOCK 38 /* Socket operation on non-socket */
|
|
||||||
#define EDESTADDRREQ 39 /* Destination address required */
|
|
||||||
#define EMSGSIZE 40 /* Message too long */
|
|
||||||
#define EPROTOTYPE 41 /* Protocol wrong type for socket */
|
|
||||||
#define ENOPROTOOPT 42 /* Protocol not available */
|
|
||||||
#define EPROTONOSUPPORT 43 /* Protocol not supported */
|
|
||||||
#define ESOCKTNOSUPPORT 44 /* Socket type not supported */
|
|
||||||
#define ENOTSUP 45 /* Operation not supported */
|
|
||||||
#define EOPNOTSUPP ENOTSUP /* Operation not supported on socket */
|
|
||||||
#define EPFNOSUPPORT 46 /* Protocol family not supported */
|
|
||||||
#define EAFNOSUPPORT 47 /* Address family not supported by protocol family */
|
|
||||||
#define EADDRINUSE 48 /* Address already in use */
|
|
||||||
#define EADDRNOTAVAIL 49 /* Can't assign requested address */
|
|
||||||
|
|
||||||
/* ipc/network software -- operational errors */
|
|
||||||
#define ENETDOWN 50 /* Network is down */
|
|
||||||
#define ENETUNREACH 51 /* Network is unreachable */
|
|
||||||
#define ENETRESET 52 /* Network dropped connection on reset */
|
|
||||||
#define ECONNABORTED 53 /* Software caused connection abort */
|
|
||||||
#define ECONNRESET 54 /* Connection reset by peer */
|
|
||||||
#define ENOBUFS 55 /* No buffer space available */
|
|
||||||
#define EISCONN 56 /* Socket is already connected */
|
|
||||||
#define ENOTCONN 57 /* Socket is not connected */
|
|
||||||
#define ESHUTDOWN 58 /* Can't send after socket shutdown */
|
|
||||||
#define ETOOMANYREFS 59 /* Too many references: can't splice */
|
|
||||||
#define ETIMEDOUT 60 /* Operation timed out */
|
|
||||||
#define ECONNREFUSED 61 /* Connection refused */
|
|
||||||
#define ELOOP 62 /* Too many levels of symbolic links */
|
|
||||||
#define ENAMETOOLONG 63 /* File name too long */
|
|
||||||
|
|
||||||
/* should be rearranged */
|
|
||||||
#define EHOSTDOWN 64 /* Host is down */
|
|
||||||
#define EHOSTUNREACH 65 /* No route to host */
|
|
||||||
#define ENOTEMPTY 66 /* Directory not empty */
|
|
||||||
|
|
||||||
/* quotas & mush */
|
|
||||||
#define EPROCLIM 67 /* Too many processes */
|
|
||||||
#define EUSERS 68 /* Too many users */
|
|
||||||
#define EDQUOT 69 /* Disc quota exceeded */
|
|
||||||
|
|
||||||
/* Network File System */
|
|
||||||
#define ESTALE 70 /* Stale NFS file handle */
|
|
||||||
#define EREMOTE 71 /* Too many levels of remote in path */
|
|
||||||
#define EBADRPC 72 /* RPC struct is bad */
|
|
||||||
#define ERPCMISMATCH 73 /* RPC version wrong */
|
|
||||||
#define EPROGUNAVAIL 74 /* RPC prog. not avail */
|
|
||||||
#define EPROGMISMATCH 75 /* Program version wrong */
|
|
||||||
#define EPROCUNAVAIL 76 /* Bad procedure for program */
|
|
||||||
#define ENOLCK 77 /* No locks available */
|
|
||||||
#define ENOSYS 78 /* Function not implemented */
|
|
||||||
#define EFTYPE 79 /* Inappropriate file type or format */
|
|
||||||
#define EAUTH 80 /* Authentication error */
|
|
||||||
#define ENEEDAUTH 81 /* Need authenticator */
|
|
||||||
|
|
||||||
/* Intelligent device errors */
|
|
||||||
#define EPWROFF 82 /* Device power is off */
|
|
||||||
#define EDEVERR 83 /* Device error, e.g. paper out */
|
|
||||||
#define EOVERFLOW 84 /* Value too large to be stored in data type */
|
|
||||||
|
|
||||||
/* Program loading errors */
|
|
||||||
#define EBADEXEC 85 /* Bad executable */
|
|
||||||
#define EBADARCH 86 /* Bad CPU type in executable */
|
|
||||||
#define ESHLIBVERS 87 /* Shared library version mismatch */
|
|
||||||
#define EBADMACHO 88 /* Malformed Macho file */
|
|
||||||
#define ECANCELED 89 /* Operation canceled */
|
|
||||||
#define EIDRM 90 /* Identifier removed */
|
|
||||||
#define ENOMSG 91 /* No message of desired type */
|
|
||||||
#define EILSEQ 92 /* Illegal byte sequence */
|
|
||||||
#define ENOATTR 93 /* Attribute not found */
|
|
||||||
#define EBADMSG 94 /* Bad message */
|
|
||||||
#define EMULTIHOP 95 /* Reserved */
|
|
||||||
#define ENODATA 96 /* No message available on STREAM */
|
|
||||||
#define ENOLINK 97 /* Reserved */
|
|
||||||
#define ENOSR 98 /* No STREAM resources */
|
|
||||||
#define ENOSTR 99 /* Not a STREAM */
|
|
||||||
#define EPROTO 100 /* Protocol error */
|
|
||||||
#define ETIME 101 /* STREAM ioctl timeout */
|
|
||||||
#define ENOPOLICY 103 /* No such policy registered */
|
|
||||||
#define ENOTRECOVERABLE 104 /* State not recoverable */
|
|
||||||
#define EOWNERDEAD 105 /* Previous owner died */
|
|
||||||
#define EQFULL 106 /* Interface output queue is full */
|
|
||||||
#define ELAST 106 /* Must be equal largest errno */
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // _ERRNO_H_
|
|
@ -1,65 +0,0 @@
|
|||||||
#ifndef _FLOAT_H_
|
|
||||||
#define _FLOAT_H_
|
|
||||||
|
|
||||||
#define FLT_RADIX 2
|
|
||||||
|
|
||||||
/* IEEE float */
|
|
||||||
#define FLT_MANT_DIG 24
|
|
||||||
#define FLT_DIG 6
|
|
||||||
#define FLT_ROUNDS 1
|
|
||||||
#define FLT_EPSILON 1.19209290e-07F
|
|
||||||
#define FLT_MIN_EXP (-125)
|
|
||||||
#define FLT_MIN 1.17549435e-38F
|
|
||||||
#define FLT_MIN_10_EXP (-37)
|
|
||||||
#define FLT_MAX_EXP 128
|
|
||||||
#define FLT_MAX 3.40282347e+38F
|
|
||||||
#define FLT_MAX_10_EXP 38
|
|
||||||
|
|
||||||
/* IEEE double */
|
|
||||||
#define DBL_MANT_DIG 53
|
|
||||||
#define DBL_DIG 15
|
|
||||||
#define DBL_EPSILON 2.2204460492503131e-16
|
|
||||||
#define DBL_MIN_EXP (-1021)
|
|
||||||
#define DBL_MIN 2.2250738585072014e-308
|
|
||||||
#define DBL_MIN_10_EXP (-307)
|
|
||||||
#define DBL_MAX_EXP 1024
|
|
||||||
#define DBL_MAX 1.7976931348623157e+308
|
|
||||||
#define DBL_MAX_10_EXP 308
|
|
||||||
|
|
||||||
/* horrible intel long double */
|
|
||||||
#if defined __i386__ || defined __x86_64__
|
|
||||||
|
|
||||||
#define LDBL_MANT_DIG 64
|
|
||||||
#define LDBL_DIG 18
|
|
||||||
#define LDBL_EPSILON 1.08420217248550443401e-19L
|
|
||||||
#define LDBL_MIN_EXP (-16381)
|
|
||||||
#define LDBL_MIN 3.36210314311209350626e-4932L
|
|
||||||
#define LDBL_MIN_10_EXP (-4931)
|
|
||||||
#define LDBL_MAX_EXP 16384
|
|
||||||
#define LDBL_MAX 1.18973149535723176502e+4932L
|
|
||||||
#define LDBL_MAX_10_EXP 4932
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
/* same as IEEE double */
|
|
||||||
#define LDBL_MANT_DIG 53
|
|
||||||
#define LDBL_DIG 15
|
|
||||||
#define LDBL_EPSILON 2.2204460492503131e-16
|
|
||||||
#define LDBL_MIN_EXP (-1021)
|
|
||||||
#define LDBL_MIN 2.2250738585072014e-308
|
|
||||||
#define LDBL_MIN_10_EXP (-307)
|
|
||||||
#define LDBL_MAX_EXP 1024
|
|
||||||
#define LDBL_MAX 1.7976931348623157e+308
|
|
||||||
#define LDBL_MAX_10_EXP 308
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef NAN
|
|
||||||
# define NAN (__nan__)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef INFINITY
|
|
||||||
# define INFINITY (__inf__)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* _FLOAT_H_ */
|
|
@ -1,991 +0,0 @@
|
|||||||
#ifndef _KSYS_H_
|
|
||||||
#define _KSYS_H_
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#define asm_inline __asm__ __volatile__
|
|
||||||
#define not_optimized __attribute__((optimize("O0")))
|
|
||||||
|
|
||||||
#define KSYS_FS_ERR_SUCCESS 0 // Success
|
|
||||||
#define KSYS_FS_ERR_1 1 // Base and/or partition of a hard disk is not defined (fn21.7 & fn21.8)
|
|
||||||
#define KSYS_FS_ERR_2 2 // Function is not supported for the given file system
|
|
||||||
#define KSYS_FS_ERR_3 3 // Unknown file system
|
|
||||||
#define KSYS_FS_ERR_4 4 // Reserved, is never returned in the current implementation
|
|
||||||
#define KSYS_FS_ERR_5 5 // File not found
|
|
||||||
#define KSYS_FS_ERR_EOF 6 // End of file, EOF
|
|
||||||
#define KSYS_FS_ERR_7 7 // Pointer lies outside of application memory
|
|
||||||
#define KSYS_FS_ERR_8 8 // Disk is full
|
|
||||||
#define KSYS_FS_ERR_9 9 // FAT table is destroyed
|
|
||||||
#define KSYS_FS_ERR_10 10 // Access denied
|
|
||||||
#define KSYS_FS_ERR_11 11 // Device error
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
unsigned char blue;
|
|
||||||
unsigned char green;
|
|
||||||
unsigned char red;
|
|
||||||
}rgb_t;
|
|
||||||
|
|
||||||
typedef unsigned int ksys_color_t;
|
|
||||||
|
|
||||||
#pragma pack(push,1)
|
|
||||||
typedef union{
|
|
||||||
unsigned val;
|
|
||||||
struct{
|
|
||||||
short x;
|
|
||||||
short y;
|
|
||||||
};
|
|
||||||
}ksys_pos_t;
|
|
||||||
|
|
||||||
typedef union ksys_oskey_t{
|
|
||||||
unsigned val;
|
|
||||||
struct{
|
|
||||||
unsigned char state;
|
|
||||||
unsigned char code;
|
|
||||||
unsigned char ctrl_key;
|
|
||||||
};
|
|
||||||
}ksys_oskey_t;
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
unsigned handle;
|
|
||||||
unsigned io_code;
|
|
||||||
unsigned *input;
|
|
||||||
int inp_size;
|
|
||||||
void *output;
|
|
||||||
int out_size;
|
|
||||||
}ksys_ioctl_t;
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
void *data;
|
|
||||||
size_t size;
|
|
||||||
}ksys_ufile_t;
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
unsigned p00;
|
|
||||||
union{
|
|
||||||
uint64_t p04;
|
|
||||||
struct {
|
|
||||||
unsigned p04dw;
|
|
||||||
unsigned p08dw;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
unsigned p12;
|
|
||||||
union {
|
|
||||||
unsigned p16;
|
|
||||||
const char *new_name;
|
|
||||||
void *bdfe;
|
|
||||||
void *buf16;
|
|
||||||
const void *cbuf16;
|
|
||||||
};
|
|
||||||
char p20;
|
|
||||||
const char *p21;
|
|
||||||
}ksys70_t;
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
unsigned attributes;
|
|
||||||
unsigned name_cp;
|
|
||||||
char creation_time[4];
|
|
||||||
char creation_date[4];
|
|
||||||
char last_access_time[4];
|
|
||||||
char last_access_date[4];
|
|
||||||
char last_modification_time[4];
|
|
||||||
char last_modification_date[4];
|
|
||||||
unsigned long long size;
|
|
||||||
char name[0];
|
|
||||||
}ksys_bdfe_t;
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
int cpu_usage; //+0
|
|
||||||
int window_pos_info; //+4
|
|
||||||
short int reserved1; //+8
|
|
||||||
char name[12]; //+10
|
|
||||||
int memstart; //+22
|
|
||||||
int memused; //+26
|
|
||||||
int pid; //+30
|
|
||||||
int winx_start; //+34
|
|
||||||
int winy_start; //+38
|
|
||||||
int winx_size; //+42
|
|
||||||
int winy_size; //+46
|
|
||||||
short int slot_info; //+50
|
|
||||||
short int reserved2; //+52
|
|
||||||
int clientx; //+54
|
|
||||||
int clienty; //+58
|
|
||||||
int clientwidth; //+62
|
|
||||||
int clientheight; //+66
|
|
||||||
unsigned char window_state;//+70
|
|
||||||
char reserved3[1024-71]; //+71
|
|
||||||
}ksys_proc_table_t;
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
ksys_color_t frame_area;
|
|
||||||
ksys_color_t grab_bar;
|
|
||||||
ksys_color_t grab_bar_button;
|
|
||||||
ksys_color_t grab_button_text;
|
|
||||||
ksys_color_t grab_text;
|
|
||||||
ksys_color_t work_area;
|
|
||||||
ksys_color_t work_button;
|
|
||||||
ksys_color_t work_button_text;
|
|
||||||
ksys_color_t work_text;
|
|
||||||
ksys_color_t work_graph;
|
|
||||||
}ksys_colors_table_t;
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
unsigned pid; // PID of sending thread
|
|
||||||
unsigned datalen; // data bytes
|
|
||||||
char *data; // data begin
|
|
||||||
}ksys_ipc_msg;
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
unsigned lock; // nonzero is locked
|
|
||||||
unsigned used; // used bytes in buffer
|
|
||||||
ksys_ipc_msg *data; // data begin
|
|
||||||
}ksys_ipc_buffer;
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
char* func_name;
|
|
||||||
void* func_ptr;
|
|
||||||
}ksys_coff_etable_t;
|
|
||||||
|
|
||||||
#pragma pack(pop)
|
|
||||||
|
|
||||||
enum KSYS_EVENTS {
|
|
||||||
KSYS_EVENT_NONE = 0, /* Event queue is empty */
|
|
||||||
KSYS_EVENT_REDRAW = 1, /* Window and window elements should be redrawn */
|
|
||||||
KSYS_EVENT_KEY = 2, /* A key on the keyboard was pressed */
|
|
||||||
KSYS_EVENT_BUTTON = 3, /* A button was clicked with the mouse */
|
|
||||||
KSYS_EVENT_DESKTOP = 5, /* Desktop redraw finished */
|
|
||||||
KSYS_EVENT_MOUSE = 6, /* Mouse activity (movement, button press) was detected */
|
|
||||||
KSYS_EVENT_IPC = 7, /* Interprocess communication notify */
|
|
||||||
KSYS_EVENT_NETWORK = 8, /* Network event */
|
|
||||||
KSYS_EVENT_DEBUG = 9, /* Debug subsystem event */
|
|
||||||
KSYS_EVENT_IRQBEGIN = 16 /* 16..31 IRQ0..IRQ15 interrupt =IRQBEGIN+IRQn */
|
|
||||||
};
|
|
||||||
|
|
||||||
enum KSYS_FILE_ENCODING{
|
|
||||||
KSYS_FILE_CP866 =1,
|
|
||||||
KSYS_FILE_UTF16LE = 2,
|
|
||||||
KSYS_FILE_UTF8 = 3
|
|
||||||
};
|
|
||||||
|
|
||||||
enum KSYS_CLIP_ENCODING{
|
|
||||||
KSYS_CLIP_UTF8 = 0,
|
|
||||||
KSYS_CLIP_CP866 = 1,
|
|
||||||
KSYS_CLIP_CP1251 = 2
|
|
||||||
};
|
|
||||||
|
|
||||||
enum KSYS_CLIP_TYPES{
|
|
||||||
KSYS_CLIP_TEXT = 0,
|
|
||||||
KSYS_CLIP_IMAGE = 1,
|
|
||||||
KSYS_CLIP_RAW = 2
|
|
||||||
};
|
|
||||||
|
|
||||||
enum KSYS_MOUSE_POS{
|
|
||||||
KSYS_MOUSE_SCREEN_POS = 0,
|
|
||||||
KSYS_MOUSE_WINDOW_POS = 1
|
|
||||||
};
|
|
||||||
|
|
||||||
enum KSYS_SHM_MODE{
|
|
||||||
KSYS_SHM_OPEN = 0x00,
|
|
||||||
KSYS_SHM_OPEN_ALWAYS = 0x04,
|
|
||||||
KSYS_SHM_CREATE = 0x08,
|
|
||||||
KSYS_SHM_READ = 0x00,
|
|
||||||
KSYS_SHM_WRITE = 0x01,
|
|
||||||
};
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_strcmp(const char * s1, const char * s2 )
|
|
||||||
{
|
|
||||||
while ((*s1) && (*s1 == *s2)){ ++s1; ++s2; }
|
|
||||||
return(*(unsigned char*)s1 - *(unsigned char *)s2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Functions for working with the graphical interface
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_start_draw()
|
|
||||||
{
|
|
||||||
asm_inline("int $0x40"::"a"(12),"b"(1));
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_end_draw()
|
|
||||||
{
|
|
||||||
asm_inline("int $0x40" ::"a"(12),"b"(2));
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_create_window(int x, int y, int w, int h, const char *name, ksys_color_t workcolor, unsigned style)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(0),
|
|
||||||
"b"((x << 16) | ((w-1) & 0xFFFF)),
|
|
||||||
"c"((y << 16) | ((h-1) & 0xFFFF)),
|
|
||||||
"d"((style << 24) | (workcolor & 0xFFFFFF)),
|
|
||||||
"D"(name),
|
|
||||||
"S"(0) : "memory"
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_change_window(int new_x, int new_y, int new_w, int new_h)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(67), "b"(new_x), "c"(new_y), "d"(new_w),"S"(new_h)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_define_button(unsigned x, unsigned y, unsigned w, unsigned h, unsigned id, ksys_color_t color)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(8),
|
|
||||||
"b"((x<<16)+w),
|
|
||||||
"c"((y<<16)+h),
|
|
||||||
"d"(id),
|
|
||||||
"S"(color)
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_line(int xs, int ys, int xe, int ye, ksys_color_t color)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(38), "d"(color),
|
|
||||||
"b"((xs << 16) | xe),
|
|
||||||
"c"((ys << 16) | ye)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_bar(int x, int y, int w, int h, ksys_color_t color)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(13), "d"(color),
|
|
||||||
"b"((x << 16) | w),
|
|
||||||
"c"((y << 16) | h)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_bitmap(void *bitmap, int x, int y, int w, int h)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(7), "b"(bitmap),
|
|
||||||
"c"((w << 16) | h),
|
|
||||||
"d"((x << 16) | y)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_text(const char *text, int x, int y, int len, ksys_color_t color)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(4),"d"(text),
|
|
||||||
"b"((x << 16) | y),
|
|
||||||
"S"(len),"c"(color)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_text_bg(const char *text, int x, int y, int len, ksys_color_t color, ksys_color_t bg)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(4),"d"(text),
|
|
||||||
"b"((x << 16) | y),
|
|
||||||
"S"(len),"c"(color), "D"(bg)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_number(int number, int x, int y, int len, ksys_color_t color){
|
|
||||||
unsigned fmt;
|
|
||||||
fmt = len << 16 | 0x80000000; // no leading zeros + width
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(47), "b"(fmt), "c"(number), "d"((x << 16) | y), "S"(color)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_draw_number_bg(unsigned number, int x, int y, int len, ksys_color_t color, ksys_color_t bg){
|
|
||||||
unsigned fmt;
|
|
||||||
fmt = len << 16 | 0x80000000; // no leading zeros + width
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(47), "b"(fmt), "c"(number), "d"((x << 16) | y), "S"(color), "D"(bg)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_skin_height()
|
|
||||||
{
|
|
||||||
unsigned height;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40 \n\t"
|
|
||||||
:"=a"(height)
|
|
||||||
:"a"(48),"b"(4)
|
|
||||||
);
|
|
||||||
return height;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_get_colors(ksys_colors_table_t *color_table)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(48),"b"(3),"c"(color_table),"d"(40)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Functions for working with a screen. */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_pos_t not_optimized _ksys_screen_size()
|
|
||||||
{
|
|
||||||
ksys_pos_t size;
|
|
||||||
ksys_pos_t size_tmp;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(size_tmp)
|
|
||||||
:"a"(14)
|
|
||||||
);
|
|
||||||
size.x = size_tmp.y;
|
|
||||||
size.y = size_tmp.x;
|
|
||||||
return size;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Functions for working with a mouse and cursors. */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_pos_t _ksys_get_mouse_pos(int origin)
|
|
||||||
{
|
|
||||||
ksys_pos_t val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40 \n\t"
|
|
||||||
"rol $16, %%eax"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(37),"b"(origin)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_mouse_buttons()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(37),"b"(2)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_mouse_wheels()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40 \n\t"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(37),"b"(7)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_load_cursor(void *path, unsigned flags)
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(37), "b"(4), "c"(path), "d"(flags)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_set_cursor(unsigned cursor)
|
|
||||||
{
|
|
||||||
unsigned old;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(old)
|
|
||||||
:"a"(37), "b"(5), "c"(cursor)
|
|
||||||
);
|
|
||||||
return old;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_destroy_cursor(unsigned cursor)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(ret)
|
|
||||||
:"a"(37), "b"(6), "c"(cursor)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_mouse_eventstate()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(37),"b"(3)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Functions for working with events and buttons. */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_set_event_mask(unsigned mask)
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(40), "b"(mask)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_wait_event(unsigned time)
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(23), "b"(time)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_check_event()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(11)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_event()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(10)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_button()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(17)
|
|
||||||
);
|
|
||||||
return val>>8;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_oskey_t _ksys_get_key(void)
|
|
||||||
{
|
|
||||||
ksys_oskey_t val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(2)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Functions for working with the clipboard */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_clip_num()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(54), "b"(0)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
char* _ksys_clip_get(int n) // returned buffer must be freed by _ksys_free()
|
|
||||||
{
|
|
||||||
char* val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(54), "b"(1), "c"(n)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_clip_set(int n, char *buffer)
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(54), "b"(2), "c"(n), "d"(buffer)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_clip_pop()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline (
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(54), "b"(3)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_clip_unlock()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(54), "b"(4)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Working with time */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_tick_count()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(26),"b"(9)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
uint64_t _ksys_get_ns_count()
|
|
||||||
{
|
|
||||||
uint64_t val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=A"(val)
|
|
||||||
:"a"(26), "b"(10)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_delay(unsigned time)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(5), "b"(time)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_date()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline("int $0x40":"=a"(val):"a"(29));
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
unsigned _ksys_get_clock()
|
|
||||||
{
|
|
||||||
unsigned val;
|
|
||||||
asm_inline("int $0x40":"=a"(val):"a"(3));
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Working with memory allocation */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void* _ksys_alloc(size_t size){
|
|
||||||
void *val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(68),"b"(12),"c"(size)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_free(void *mem)
|
|
||||||
{
|
|
||||||
int val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(68),"b"(13),"c"(mem)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void* _ksys_realloc(void *mem, size_t size)
|
|
||||||
{
|
|
||||||
void *val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(68),"b"(20),"c"(size),"d"(mem)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int* _ksys_unmap(void *base, size_t offset, size_t size)
|
|
||||||
{
|
|
||||||
int *val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(68),"b"(26),"c"(base),"d"(offset),"S"(size)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Loading the dynamic coff library */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_coff_etable_t* not_optimized _ksys_cofflib_load(const char* path)
|
|
||||||
{
|
|
||||||
ksys_coff_etable_t *table;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(table)
|
|
||||||
:"a"(68),"b"(19), "c"(path)
|
|
||||||
);
|
|
||||||
return table;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void* not_optimized _ksys_cofflib_getproc(ksys_coff_etable_t *table, const char* fun_name)
|
|
||||||
{
|
|
||||||
unsigned i=0;
|
|
||||||
while (1){
|
|
||||||
if (NULL == (table+i)->func_name){
|
|
||||||
break;
|
|
||||||
}else{
|
|
||||||
if (!_ksys_strcmp(fun_name, (table+i)->func_name)){
|
|
||||||
return (table+i)->func_ptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Debug board functions */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_debug_putc(char c)
|
|
||||||
{
|
|
||||||
asm_inline("int $0x40"::"a"(63), "b"(1), "c"(c));
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_debug_puts(char *s)
|
|
||||||
{
|
|
||||||
unsigned i=0;
|
|
||||||
while (*(s+i)){
|
|
||||||
asm_inline ("int $0x40"::"a"(63), "b"(1), "c"(*(s+i)));
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Working with threads and process */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_start_thread(void* proc, char* stack_top)
|
|
||||||
{
|
|
||||||
int val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(51), "b"(1), "c"(proc), "d"(stack_top)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_focus_window(int slot){
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(18), "b"(3), "c"(slot)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_get_thread_slot(int tid){
|
|
||||||
int val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(18), "b"(21), "c"(tid)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_process_info(ksys_proc_table_t* table, int pid)
|
|
||||||
{
|
|
||||||
int val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val)
|
|
||||||
:"a"(9), "b"(table), "c"(pid)
|
|
||||||
:"memory"
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_exit()
|
|
||||||
{
|
|
||||||
asm_inline("int $0x40"::"a"(-1));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Working with files and directories */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_setcwd(char* dir){
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
::"a"(30), "b"(1), "c"(dir)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_getcwd(char* buf, int bufsize){
|
|
||||||
register int val;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(val):"a"(30), "b"(2), "c"(buf), "d"(bufsize)
|
|
||||||
);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_ufile_t _ksys_load_file(const char *path)
|
|
||||||
{
|
|
||||||
ksys_ufile_t uf;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(uf.data), "=d"(uf.size)
|
|
||||||
:"a"(68), "b"(27),"c"(path)
|
|
||||||
);
|
|
||||||
return uf;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
ksys_ufile_t _ksys_load_file_enc(const char *path, unsigned file_encoding)
|
|
||||||
{
|
|
||||||
ksys_ufile_t uf;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(uf.data), "=d"(uf.size)
|
|
||||||
:"a"(68), "b"(28),"c"(path), "d"(file_encoding)
|
|
||||||
);
|
|
||||||
return uf;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_work_files(const ksys70_t *k)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(status)
|
|
||||||
:"a"(70), "b"(k)
|
|
||||||
);
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_file_read_file(const char *name, unsigned long long offset, unsigned size, void *buf, unsigned *bytes_read)
|
|
||||||
{
|
|
||||||
ksys70_t k;
|
|
||||||
k.p00 = 0;
|
|
||||||
k.p04 = offset;
|
|
||||||
k.p12 = size;
|
|
||||||
k.buf16 = buf;
|
|
||||||
k.p20 = 0;
|
|
||||||
k.p21 = name;
|
|
||||||
int status;
|
|
||||||
unsigned bytes_read_v;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(status), "=b"(bytes_read_v)
|
|
||||||
:"a"(70), "b"(&k)
|
|
||||||
);
|
|
||||||
if (!status) {
|
|
||||||
*bytes_read = bytes_read_v;
|
|
||||||
}
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_file_write_file(const char *name, unsigned long long offset, unsigned size, const void *buf, unsigned *bytes_written)
|
|
||||||
{
|
|
||||||
ksys70_t k;
|
|
||||||
k.p00 = 3;
|
|
||||||
k.p04 = offset;
|
|
||||||
k.p12 = size;
|
|
||||||
k.cbuf16 = buf;
|
|
||||||
k.p20 = 0;
|
|
||||||
k.p21 = name;
|
|
||||||
int status;
|
|
||||||
unsigned bytes_written_v;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(status), "=b"(bytes_written_v)
|
|
||||||
:"a"(70), "b"(&k)
|
|
||||||
);
|
|
||||||
if (!status) {
|
|
||||||
*bytes_written = bytes_written_v;
|
|
||||||
}
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_file_get_info(const char *name, ksys_bdfe_t *bdfe)
|
|
||||||
{
|
|
||||||
ksys70_t k;
|
|
||||||
k.p00 = 5;
|
|
||||||
k.bdfe = bdfe;
|
|
||||||
k.p20 = 0;
|
|
||||||
k.p21 = name;
|
|
||||||
return _ksys_work_files(&k);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_file_delete(const char *name)
|
|
||||||
{
|
|
||||||
ksys70_t k;
|
|
||||||
k.p00 = 8;
|
|
||||||
k.p20 = 0;
|
|
||||||
k.p21 = name;
|
|
||||||
return _ksys_work_files(&k);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_file_rename(const char *name, const char *new_name)
|
|
||||||
{
|
|
||||||
ksys70_t k;
|
|
||||||
k.p00 = 10;
|
|
||||||
k.new_name = new_name;
|
|
||||||
k.p20 = 0;
|
|
||||||
k.p21 = name;
|
|
||||||
return _ksys_work_files(&k);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_exec(char *app_name, char *args)
|
|
||||||
{
|
|
||||||
ksys70_t file_opt;
|
|
||||||
file_opt.p00 = 7;
|
|
||||||
file_opt.p04dw = 0;
|
|
||||||
file_opt.p08dw = (unsigned)args;
|
|
||||||
file_opt.p21 = app_name;
|
|
||||||
return _ksys_work_files(&file_opt);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int not_optimized _ksys_mkdir(const char *path)
|
|
||||||
{
|
|
||||||
ksys70_t dir_opt;
|
|
||||||
dir_opt.p00 = 9;
|
|
||||||
dir_opt.p21 = path;
|
|
||||||
return _ksys_work_files(&dir_opt);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Working with a named shared memory area. */
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int _ksys_shm_open(char *name, int mode, int size, char **new_shm)
|
|
||||||
{
|
|
||||||
int error;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=a"(*new_shm), "=d"(error)
|
|
||||||
:"a"(68), "b"(22), "c"(name), "d"(size), "S"(mode)
|
|
||||||
);
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _ksys_shm_close(char *shm_name)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40":
|
|
||||||
:"a"(68), "b"(23), "c"(shm_name)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // _KSYS_H_
|
|
@ -1,18 +0,0 @@
|
|||||||
#ifndef _LIMITS_H_
|
|
||||||
#define _LIMITS_H_
|
|
||||||
|
|
||||||
|
|
||||||
#define INT_MAX 2147483647
|
|
||||||
#define UINT_MAX (INT_MAX * 2U + 1)
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef ARG_MAX
|
|
||||||
#define ARG_MAX 4096
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef PATH_MAX
|
|
||||||
#define PATH_MAX 4096
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#endif /* _LIMITS_H_ */
|
|
@ -1,171 +0,0 @@
|
|||||||
/* Copyright (C) 1999 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
/* Copyright (C) 1998 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#ifndef _MATH_H_
|
|
||||||
#define _MATH_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern double _FUNC(acos)(double _x);
|
|
||||||
extern double _FUNC(asin)(double _x);
|
|
||||||
extern double _FUNC(atan)(double _x);
|
|
||||||
extern double _FUNC(atan2)(double _y, double _x);
|
|
||||||
extern double _FUNC(ceil)(double _x);
|
|
||||||
extern double _FUNC(cos)(double _x);
|
|
||||||
extern double _FUNC(cosh)(double _x);
|
|
||||||
extern double _FUNC(exp)(double _x);
|
|
||||||
extern double _FUNC(fabs)(double _x);
|
|
||||||
extern double _FUNC(floor)(double _x);
|
|
||||||
extern double _FUNC(fmod)(double _x, double _y);
|
|
||||||
extern double _FUNC(frexp)(double _x, int *_pexp);
|
|
||||||
extern double _FUNC(ldexp)(double _x, int _exp);
|
|
||||||
extern double _FUNC(log)(double _y);
|
|
||||||
extern double _FUNC(log10)(double _x);
|
|
||||||
extern double _FUNC(modf)(double _x, double *_pint);
|
|
||||||
extern double _FUNC(pow)(double _x, double _y);
|
|
||||||
extern double _FUNC(sin)(double _x);
|
|
||||||
extern double _FUNC(sinh)(double _x);
|
|
||||||
extern double _FUNC(sqrt)(double _x);
|
|
||||||
extern double _FUNC(tan)(double _x);
|
|
||||||
extern double _FUNC(tanh)(double _x);
|
|
||||||
|
|
||||||
#define M_E 2.7182818284590452354
|
|
||||||
#define M_LOG2E 1.4426950408889634074
|
|
||||||
#define M_LOG10E 0.43429448190325182765
|
|
||||||
#define M_LN2 0.69314718055994530942
|
|
||||||
#define M_LN10 2.30258509299404568402
|
|
||||||
#define M_PI 3.14159265358979323846
|
|
||||||
#define M_PI_2 1.57079632679489661923
|
|
||||||
#define M_PI_4 0.78539816339744830962
|
|
||||||
#define M_1_PI 0.31830988618379067154
|
|
||||||
#define M_2_PI 0.63661977236758134308
|
|
||||||
#define M_2_SQRTPI 1.12837916709551257390
|
|
||||||
#define M_SQRT2 1.41421356237309504880
|
|
||||||
#define M_SQRT1_2 0.70710678118654752440
|
|
||||||
#define PI M_PI
|
|
||||||
#define PI2 M_PI_2
|
|
||||||
|
|
||||||
extern double _FUNC(acosh)(double);
|
|
||||||
extern double _FUNC(asinh)(double);
|
|
||||||
extern double _FUNC(atanh)(double);
|
|
||||||
extern double _FUNC(cbrt)(double);
|
|
||||||
extern double _FUNC(exp10)(double _x);
|
|
||||||
extern double _FUNC(exp2)(double _x);
|
|
||||||
extern double _FUNC(expm1)(double);
|
|
||||||
extern double _FUNC(hypot)(double, double);
|
|
||||||
extern double _FUNC(log1p)(double);
|
|
||||||
extern double _FUNC(log2)(double _x);
|
|
||||||
extern long double _FUNC(modfl)(long double _x, long double *_pint);
|
|
||||||
extern double _FUNC(pow10)(double _x);
|
|
||||||
extern double _FUNC(pow2)(double _x);
|
|
||||||
extern double _FUNC(powi)(double, int);
|
|
||||||
extern void _FUNC(sincos)(double, double *, double *);
|
|
||||||
|
|
||||||
/* These are in libm.a (Cygnus). You must link -lm to get these */
|
|
||||||
/* See libm/math.h for comments */
|
|
||||||
|
|
||||||
#ifndef __cplusplus
|
|
||||||
struct exception {
|
|
||||||
int type;
|
|
||||||
const char *name;
|
|
||||||
double arg1;
|
|
||||||
double arg2;
|
|
||||||
double retval;
|
|
||||||
int err;
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
extern double _FUNC(erf)(double);
|
|
||||||
extern double _FUNC(erfc)(double);
|
|
||||||
extern double _FUNC(gamma)(double);
|
|
||||||
extern int _FUNC(isinf)(double);
|
|
||||||
extern int _FUNC(isnan)(double);
|
|
||||||
extern int _FUNC(finite)(double);
|
|
||||||
extern double _FUNC(j0)(double);
|
|
||||||
extern double _FUNC(j1)(double);
|
|
||||||
extern double _FUNC(jn)(int, double);
|
|
||||||
extern double _FUNC(lgamma)(double);
|
|
||||||
extern double _FUNC(nan)(const char*);
|
|
||||||
extern double _FUNC(y0)(double);
|
|
||||||
extern double _FUNC(y1)(double);
|
|
||||||
extern double _FUNC(yn)(int, double);
|
|
||||||
extern double _FUNC(logb)(double);
|
|
||||||
extern double _FUNC(nextafter)(double, double);
|
|
||||||
extern double _FUNC(remainder)(double, double);
|
|
||||||
extern double _FUNC(scalb)(double, double);
|
|
||||||
#ifndef __cplusplus
|
|
||||||
extern int _FUNC(matherr)(struct exception *);
|
|
||||||
#endif
|
|
||||||
extern double _FUNC(significand)(double);
|
|
||||||
extern double _FUNC(copysign)(double, double);
|
|
||||||
extern int _FUNC(ilogb)(double);
|
|
||||||
extern double _FUNC(rint)(double);
|
|
||||||
extern double _FUNC(scalbn)(double, int);
|
|
||||||
extern double _FUNC(drem)(double, double);
|
|
||||||
extern double _FUNC(gamma_r)(double, int *);
|
|
||||||
extern double _FUNC(lgamma_r)(double, int *);
|
|
||||||
extern float _FUNC(acosf)(float);
|
|
||||||
extern float _FUNC(asinf)(float);
|
|
||||||
extern float _FUNC(atanf)(float);
|
|
||||||
extern float _FUNC(atan2f)(float, float);
|
|
||||||
extern float _FUNC(cosf)(float);
|
|
||||||
extern float _FUNC(sinf)(float);
|
|
||||||
extern float _FUNC(tanf)(float);
|
|
||||||
extern float _FUNC(coshf)(float);
|
|
||||||
extern float _FUNC(sinhf)(float);
|
|
||||||
extern float _FUNC(tanhf)(float);
|
|
||||||
extern float _FUNC(expf)(float);
|
|
||||||
extern float _FUNC(frexpf)(float, int *);
|
|
||||||
extern float _FUNC(ldexpf)(float, int);
|
|
||||||
extern float _FUNC(logf)(float);
|
|
||||||
extern float _FUNC(log10f)(float);
|
|
||||||
extern float _FUNC(modff)(float, float *);
|
|
||||||
extern float _FUNC(powf)(float, float);
|
|
||||||
extern float _FUNC(sqrtf)(float);
|
|
||||||
extern float _FUNC(ceilf)(float);
|
|
||||||
extern float _FUNC(fabsf)(float);
|
|
||||||
extern float _FUNC(floorf)(float);
|
|
||||||
extern float _FUNC(fmodf)(float, float);
|
|
||||||
extern float _FUNC(erff)(float);
|
|
||||||
extern float _FUNC(erfcf)(float);
|
|
||||||
extern float _FUNC(gammaf)(float);
|
|
||||||
extern float _FUNC(hypotf)(float, float);
|
|
||||||
extern int _FUNC(isinff)(float);
|
|
||||||
extern int _FUNC(isnanf)(float);
|
|
||||||
extern int _FUNC(finitef)(float);
|
|
||||||
extern float _FUNC(j0f)(float);
|
|
||||||
extern float _FUNC(j1f)(float);
|
|
||||||
extern float _FUNC(jnf)(int, float);
|
|
||||||
extern float _FUNC(lgammaf)(float);
|
|
||||||
extern float _FUNC(nanf)(const char*);
|
|
||||||
extern float _FUNC(y0f)(float);
|
|
||||||
extern float _FUNC(y1f)(float);
|
|
||||||
extern float _FUNC(ynf)(int, float);
|
|
||||||
extern float _FUNC(acoshf)(float);
|
|
||||||
extern float _FUNC(asinhf)(float);
|
|
||||||
extern float _FUNC(atanhf)(float);
|
|
||||||
extern float _FUNC(cbrtf)(float);
|
|
||||||
extern float _FUNC(logbf)(float);
|
|
||||||
extern float _FUNC(nextafterf)(float, float);
|
|
||||||
extern float _FUNC(remainderf)(float, float);
|
|
||||||
extern float _FUNC(scalbf)(float, float);
|
|
||||||
extern float _FUNC(significandf)(float);
|
|
||||||
extern float _FUNC(copysignf)(float, float);
|
|
||||||
extern int _FUNC(ilogbf)(float);
|
|
||||||
extern float _FUNC(rintf)(float);
|
|
||||||
extern float _FUNC(scalbnf)(float, int);
|
|
||||||
extern float _FUNC(dremf)(float, float);
|
|
||||||
extern float _FUNC(expm1f)(float);
|
|
||||||
extern float _FUNC(log1pf)(float);
|
|
||||||
extern float _FUNC(gammaf_r)(float, int *);
|
|
||||||
extern float _FUNC(lgammaf_r)(float, int *);
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* _MATH_H_ */
|
|
@ -1,17 +0,0 @@
|
|||||||
#ifndef _SETJMP_H_
|
|
||||||
#define _SETJMP_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
typedef unsigned long __jmp_buf[6];
|
|
||||||
|
|
||||||
typedef struct __jmp_buf_tag {
|
|
||||||
__jmp_buf __jb;
|
|
||||||
unsigned long __fl;
|
|
||||||
unsigned long __ss[128/sizeof(long)];
|
|
||||||
} jmp_buf[1];
|
|
||||||
|
|
||||||
extern int _FUNC(setjmp)(jmp_buf env);
|
|
||||||
extern void _FUNC(longjmp)(jmp_buf env, int val);
|
|
||||||
|
|
||||||
#endif // _SETJMP_H_
|
|
@ -1,15 +0,0 @@
|
|||||||
#ifndef _SHELL_API_H_
|
|
||||||
#define _SHELL_API_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
extern void _FUNC(shell_printf)(const char* format, ...);
|
|
||||||
extern void _FUNC(shell_puts)(const char *s);
|
|
||||||
extern void _FUNC(shell_putc)(char c);
|
|
||||||
extern char _FUNC(shell_getc)();
|
|
||||||
extern void _FUNC(shell_gets)(char *str);
|
|
||||||
extern void _FUNC(shell_cls)();
|
|
||||||
extern void _FUNC(shell_exit)();
|
|
||||||
extern unsigned _FUNC(shell_get_pid)();
|
|
||||||
extern int _FUNC(shell_ping)();
|
|
||||||
#endif
|
|
@ -1,77 +0,0 @@
|
|||||||
#ifndef _STDARG_H_
|
|
||||||
#define _STDARG_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#ifdef __x86_64__
|
|
||||||
#ifndef _WIN64
|
|
||||||
|
|
||||||
//This should be in sync with the declaration on our lib/libtcc1.c
|
|
||||||
/* GCC compatible definition of va_list. */
|
|
||||||
typedef struct {
|
|
||||||
unsigned int gp_offset;
|
|
||||||
unsigned int fp_offset;
|
|
||||||
union {
|
|
||||||
unsigned int overflow_offset;
|
|
||||||
char *overflow_arg_area;
|
|
||||||
};
|
|
||||||
char *reg_save_area;
|
|
||||||
} __va_list_struct;
|
|
||||||
|
|
||||||
typedef __va_list_struct va_list[1];
|
|
||||||
|
|
||||||
extern void _FUNC(__va_start)(__va_list_struct *ap, void *fp);
|
|
||||||
extern void* _FUNC(__va_arg)(__va_list_struct *ap, int arg_type, int size, int align);
|
|
||||||
|
|
||||||
#define va_start(ap, last) __va_start(ap, __builtin_frame_address(0))
|
|
||||||
#define va_arg(ap, type) \
|
|
||||||
(*(type *)(__va_arg(ap, __builtin_va_arg_types(type), sizeof(type), __alignof__(type))))
|
|
||||||
#define va_copy(dest, src) (*(dest) = *(src))
|
|
||||||
#define va_end(ap)
|
|
||||||
|
|
||||||
#else /* _WIN64 */
|
|
||||||
typedef char *va_list;
|
|
||||||
#define va_start(ap,last) __builtin_va_start(ap,last)
|
|
||||||
#define va_arg(ap,type) (ap += 8, sizeof(type)<=8 ? *(type*)ap : **(type**)ap)
|
|
||||||
#define va_copy(dest, src) ((dest) = (src))
|
|
||||||
#define va_end(ap)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#elif __arm__
|
|
||||||
typedef char *va_list;
|
|
||||||
#define _tcc_alignof(type) ((int)&((struct {char c;type x;} *)0)->x)
|
|
||||||
#define _tcc_align(addr,type) (((unsigned)addr + _tcc_alignof(type) - 1) \
|
|
||||||
& ~(_tcc_alignof(type) - 1))
|
|
||||||
#define va_start(ap,last) ap = ((char *)&(last)) + ((sizeof(last)+3)&~3)
|
|
||||||
#define va_arg(ap,type) (ap = (void *) ((_tcc_align(ap,type)+sizeof(type)+3) \
|
|
||||||
&~3), *(type *)(ap - ((sizeof(type)+3)&~3)))
|
|
||||||
#define va_copy(dest, src) (dest) = (src)
|
|
||||||
#define va_end(ap)
|
|
||||||
|
|
||||||
#elif defined(__aarch64__)
|
|
||||||
typedef struct {
|
|
||||||
void *__stack;
|
|
||||||
void *__gr_top;
|
|
||||||
void *__vr_top;
|
|
||||||
int __gr_offs;
|
|
||||||
int __vr_offs;
|
|
||||||
} va_list;
|
|
||||||
#define va_start(ap, last) __va_start(ap, last)
|
|
||||||
#define va_arg(ap, type) __va_arg(ap, type)
|
|
||||||
#define va_end(ap)
|
|
||||||
#define va_copy(dest, src) ((dest) = (src))
|
|
||||||
|
|
||||||
#else /* __i386__ */
|
|
||||||
typedef char *va_list;
|
|
||||||
/* only correct for i386 */
|
|
||||||
#define va_start(ap,last) ap = ((char *)&(last)) + ((sizeof(last)+3)&~3)
|
|
||||||
#define va_arg(ap,type) (ap += (sizeof(type)+3)&~3, *(type *)(ap - ((sizeof(type)+3)&~3)))
|
|
||||||
#define va_copy(dest, src) (dest) = (src)
|
|
||||||
#define va_end(ap)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* fix a buggy dependency on GCC in libio.h */
|
|
||||||
typedef va_list __gnuc_va_list;
|
|
||||||
#define _VA_LIST_DEFINED
|
|
||||||
|
|
||||||
#endif /* _STDARG_H */
|
|
@ -1,11 +0,0 @@
|
|||||||
#ifndef _STDBOOL_H_
|
|
||||||
#define _STDBOOL_H_
|
|
||||||
|
|
||||||
/* ISOC99 boolean */
|
|
||||||
|
|
||||||
#define bool _Bool
|
|
||||||
#define true 1
|
|
||||||
#define false 0
|
|
||||||
#define __bool_true_false_are_defined 1
|
|
||||||
|
|
||||||
#endif /* _STDBOOL_H */
|
|
@ -1,39 +0,0 @@
|
|||||||
#ifndef _STDDEF_H_
|
|
||||||
#define _STDDEF_H_
|
|
||||||
|
|
||||||
typedef __SIZE_TYPE__ size_t;
|
|
||||||
typedef __PTRDIFF_TYPE__ ssize_t;
|
|
||||||
typedef __WCHAR_TYPE__ wchar_t;
|
|
||||||
typedef __PTRDIFF_TYPE__ ptrdiff_t;
|
|
||||||
typedef __PTRDIFF_TYPE__ intptr_t;
|
|
||||||
typedef __SIZE_TYPE__ uintptr_t;
|
|
||||||
|
|
||||||
#ifndef __int8_t_defined
|
|
||||||
#define __int8_t_defined
|
|
||||||
typedef signed char int8_t;
|
|
||||||
typedef signed short int int16_t;
|
|
||||||
typedef signed int int32_t;
|
|
||||||
typedef signed long long int int64_t;
|
|
||||||
typedef unsigned char uint8_t;
|
|
||||||
typedef unsigned short int uint16_t;
|
|
||||||
typedef unsigned int uint32_t;
|
|
||||||
typedef unsigned long long int uint64_t;
|
|
||||||
typedef int64_t intmax_t;
|
|
||||||
typedef uint64_t uintmax_t;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef NULL
|
|
||||||
#define NULL ((void*)0)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _DYNAMIC
|
|
||||||
#define _FUNC(func) (*func)
|
|
||||||
#define _GLOBAL_VAR(var) *var
|
|
||||||
#else
|
|
||||||
#define _FUNC(func) func
|
|
||||||
#define _GLOBAL_VAR(var) var
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define offsetof(type, field) ((size_t)&((type *)0)->field)
|
|
||||||
|
|
||||||
#endif /* _STDDEF_H_ */
|
|
@ -1,28 +0,0 @@
|
|||||||
#ifndef _STDINT_H_
|
|
||||||
#define _STDINT_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#define INT8_MIN (-128)
|
|
||||||
#define INT8_MAX (127)
|
|
||||||
#define UINT8_MAX (255)
|
|
||||||
|
|
||||||
#define INT16_MIN (-32768)
|
|
||||||
#define INT16_MAX (32767)
|
|
||||||
#define UINT16_MAX (65535)
|
|
||||||
|
|
||||||
#define INT32_MIN (-2147483647L-1)
|
|
||||||
#define INT32_MAX (2147483647L)
|
|
||||||
#define UINT32_MAX (4294967295UL)
|
|
||||||
|
|
||||||
#if __have_long64
|
|
||||||
#define INT64_MIN (-9223372036854775807L-1L)
|
|
||||||
#define INT64_MAX (9223372036854775807L)
|
|
||||||
#define UINT64_MAX (18446744073709551615U)
|
|
||||||
#elif __have_longlong64
|
|
||||||
#define INT64_MIN (-9223372036854775807LL-1LL)
|
|
||||||
#define INT64_MAX (9223372036854775807LL)
|
|
||||||
#define UINT64_MAX (18446744073709551615ULL)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* _STDINT_H_*/
|
|
@ -1,143 +0,0 @@
|
|||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// \author (c) Marco Paland (info@paland.com)
|
|
||||||
// 2014-2019, PALANDesign Hannover, Germany
|
|
||||||
//
|
|
||||||
// \license The MIT License (MIT)
|
|
||||||
//
|
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
|
||||||
// in the Software without restriction, including without limitation the rights
|
|
||||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
// copies of the Software, and to permit persons to whom the Software is
|
|
||||||
// furnished to do so, subject to the following conditions:
|
|
||||||
//
|
|
||||||
// The above copyright notice and this permission notice shall be included in
|
|
||||||
// all copies or substantial portions of the Software.
|
|
||||||
//
|
|
||||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
// THE SOFTWARE.
|
|
||||||
//
|
|
||||||
// \brief Tiny printf, sprintf and snprintf implementation, optimized for speed on
|
|
||||||
// embedded systems with a very limited resources.
|
|
||||||
// Use this instead of bloated standard/newlib printf.
|
|
||||||
// These routines are thread safe and reentrant.
|
|
||||||
//
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#ifndef _STDIO_H_
|
|
||||||
#define _STDIO_H_
|
|
||||||
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <stddef.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
extern int _FUNC(puts)(const char *str);
|
|
||||||
extern int _FUNC(printf)(const char* format, ...);
|
|
||||||
extern int _FUNC(sprintf)(char* buffer, const char* format, ...);
|
|
||||||
extern int _FUNC(snprintf)(char* buffer, size_t count, const char* format, ...);
|
|
||||||
extern int _FUNC(vsnprintf)(char* buffer, size_t count, const char* format, va_list va);
|
|
||||||
extern int _FUNC(vprintf)(const char* format, va_list va);
|
|
||||||
|
|
||||||
extern void _FUNC(debug_printf)(const char* format, ...);
|
|
||||||
|
|
||||||
typedef size_t fpos_t;
|
|
||||||
|
|
||||||
#define _STDIO_F_R 1 << 0 // Read
|
|
||||||
#define _STDIO_F_W 1 << 1 // Write
|
|
||||||
#define _STDIO_F_A 1 << 2 // Append
|
|
||||||
#define _STDIO_F_X 1 << 3 // eXclusive
|
|
||||||
#define _STDIO_F_B 1 << 4 // Binary
|
|
||||||
|
|
||||||
typedef struct FILE_s {
|
|
||||||
char *name;
|
|
||||||
fpos_t position;
|
|
||||||
int error;
|
|
||||||
int eof;
|
|
||||||
int kind; // 0 - undiefned, 1 - text, 2 - binary
|
|
||||||
int orientation; // 0 - undiefned, 1 - byte, 2 - wide
|
|
||||||
int mode; // flags _STDIO_F_*
|
|
||||||
int append_offset; // do not seek before this point ("a" mode)
|
|
||||||
int __ungetc_emu_buff; // Uses __ungetc_emu (temporary solution!)
|
|
||||||
} FILE;
|
|
||||||
|
|
||||||
#define _IOFBF 0
|
|
||||||
#define _IOLBF 1
|
|
||||||
#define _IONBF 2
|
|
||||||
|
|
||||||
#define BUFSIZ 1024
|
|
||||||
|
|
||||||
#define EOF -1
|
|
||||||
|
|
||||||
#define FOPEN_MAX 0xffffffff
|
|
||||||
|
|
||||||
#define FILENAME_MAX 255
|
|
||||||
|
|
||||||
#define L_tmpnam FILENAME_MAX
|
|
||||||
|
|
||||||
#define SEEK_CUR 0
|
|
||||||
#define SEEK_END 1
|
|
||||||
#define SEEK_SET 2
|
|
||||||
|
|
||||||
#define TMP_MAX FOPEN_MAX
|
|
||||||
|
|
||||||
#define stderr 0is7ye
|
|
||||||
#define stdin 8yfg8e
|
|
||||||
#define stdout 7hdgys
|
|
||||||
|
|
||||||
extern int _FUNC(fgetc)(FILE *);
|
|
||||||
extern char* _FUNC(fgets)(char *restrict, int, FILE *restrict);
|
|
||||||
extern int _FUNC(fprintf)(FILE *restrict, const char *restrict, ...);
|
|
||||||
extern int _FUNC(fputc)(int, FILE *);
|
|
||||||
extern int _FUNC(fputs)(const char *restrict, FILE *restrict);
|
|
||||||
extern size_t _FUNC(fread)(void *restrict, size_t size, size_t count, FILE *restrict);
|
|
||||||
extern int _FUNC(fscanf)(FILE *restrict, const char *restrict, ...);
|
|
||||||
extern size_t _FUNC(fwrite)(const void *restrict, size_t size, size_t count, FILE *restrict);
|
|
||||||
extern int _FUNC(getc)(FILE *);
|
|
||||||
#define getc _FUNC(fgetc)
|
|
||||||
extern int _FUNC(getchar)(void);
|
|
||||||
extern int _FUNC(printf)(const char *restrict, ...);
|
|
||||||
extern int _FUNC(putc)(int, FILE *);
|
|
||||||
extern int _FUNC(putchar)(int);
|
|
||||||
extern int _FUNC(puts)(const char *);
|
|
||||||
extern int _FUNC(scanf)(const char *restrict, ...);
|
|
||||||
extern char* _FUNC(gets)(char *str);
|
|
||||||
//extern int _FUNC(ungetc)(int, FILE *);
|
|
||||||
extern int _FUNC(vfprintf)(FILE *restrict, const char *restrict, va_list);
|
|
||||||
extern int _FUNC(vfscanf)(FILE *restrict, const char *restrict, va_list);
|
|
||||||
extern int _FUNC(vprintf)(const char *restrict, va_list);
|
|
||||||
extern int _FUNC(vscanf)(const char *restrict, va_list);
|
|
||||||
extern int _FUNC(vsscanf)(const char *, const char*, va_list);
|
|
||||||
|
|
||||||
extern int _FUNC(remove)(const char *);
|
|
||||||
extern int _FUNC(rename)(const char *, const char *);
|
|
||||||
extern FILE* _FUNC(tmpfile)(void);
|
|
||||||
extern char* _FUNC(tmpnam)(char *);
|
|
||||||
|
|
||||||
extern int _FUNC(fclose)(FILE *);
|
|
||||||
extern int _FUNC(fflush)(FILE *);
|
|
||||||
extern FILE* _FUNC(fopen)(const char *restrict, const char *restrict);
|
|
||||||
extern FILE* _FUNC(freopen)(const char *restrict, const char *restrict, FILE *restrict);
|
|
||||||
extern void _FUNC(setbuf)(FILE *restrict, char *restrict);
|
|
||||||
extern int _FUNC(setvbuf)(FILE *restrict, char *restrict, int, size_t);
|
|
||||||
|
|
||||||
extern int _FUNC(fgetpos)(FILE *restrict, fpos_t *restrict);
|
|
||||||
extern int _FUNC(fseek)(FILE *, long, int);
|
|
||||||
extern int _FUNC(fsetpos)(FILE *, const fpos_t *);
|
|
||||||
extern long _FUNC(ftell)(FILE *);
|
|
||||||
extern void _FUNC(rewind)(FILE *);
|
|
||||||
|
|
||||||
extern void _FUNC(clearerr)(FILE *);
|
|
||||||
extern int _FUNC(feof)(FILE *);
|
|
||||||
extern int _FUNC(ferror)(FILE *);
|
|
||||||
extern void _FUNC(perror)(const char *);
|
|
||||||
|
|
||||||
extern size_t _FUNC(fread)(void *restrict, size_t, size_t, FILE *restrict);
|
|
||||||
|
|
||||||
extern int _FUNC(getchar)(void);
|
|
||||||
|
|
||||||
#endif // _STDIO_H_
|
|
@ -1,38 +0,0 @@
|
|||||||
#ifndef _STDLIB_H_
|
|
||||||
#define _STDLIB_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#define RAND_MAX 65535
|
|
||||||
#ifndef NULL
|
|
||||||
# define NULL ((void*)0)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define min(a, b) ((a)<(b) ? (a) : (b))
|
|
||||||
#define max(a, b) ((a)>(b) ? (a) : (b))
|
|
||||||
|
|
||||||
extern int _FUNC(atoi)(const char *s);
|
|
||||||
extern long _FUNC(atol)(const char *);
|
|
||||||
extern long long _FUNC(atoll)(const char *);
|
|
||||||
extern void _FUNC(itoa)(int n, char* s);
|
|
||||||
|
|
||||||
extern int _FUNC(abs)(int);
|
|
||||||
extern long _FUNC(labs)(long);
|
|
||||||
extern long long _FUNC(llabs)(long long);
|
|
||||||
|
|
||||||
typedef struct { int quot, rem; } div_t;
|
|
||||||
typedef struct { long quot, rem; } ldiv_t;
|
|
||||||
typedef struct { long long quot, rem; } lldiv_t;
|
|
||||||
|
|
||||||
extern div_t _FUNC(div)(int, int);
|
|
||||||
extern ldiv_t _FUNC(ldiv)(long, long);
|
|
||||||
extern lldiv_t _FUNC(lldiv)(long long, long long);
|
|
||||||
|
|
||||||
extern void* _FUNC(malloc)(size_t size);
|
|
||||||
extern void* _FUNC(calloc)(size_t num, size_t size);
|
|
||||||
extern void* _FUNC(realloc)(void *ptr, size_t newsize);
|
|
||||||
extern void _FUNC(free)(void *ptr);
|
|
||||||
|
|
||||||
extern void _FUNC(exit)(int status);
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,186 +0,0 @@
|
|||||||
/* String handling <string.h>
|
|
||||||
|
|
||||||
This file is part of the Public Domain C Library (PDCLib).
|
|
||||||
Permission is granted to use, modify, and / or redistribute at will.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _STRING_H_
|
|
||||||
#define _STRING_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* String function conventions */
|
|
||||||
|
|
||||||
/*
|
|
||||||
In any of the following functions taking a size_t n to specify the length of
|
|
||||||
an array or size of a memory region, n may be 0, but the pointer arguments to
|
|
||||||
the call shall still be valid unless otherwise stated.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* Copying functions */
|
|
||||||
|
|
||||||
extern void* _FUNC(memccpy)(void *restrict dest, const void *restrict src, int c, size_t n);
|
|
||||||
|
|
||||||
/* Copy a number of n characters from the memory area pointed to by s2 to the
|
|
||||||
area pointed to by s1. If the two areas overlap, behaviour is undefined.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern void* _FUNC(memcpy)(void* s1, const void* s2, size_t n);
|
|
||||||
|
|
||||||
/* Copy a number of n characters from the memory area pointed to by s2 to the
|
|
||||||
area pointed to by s1. The two areas may overlap.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern void* _FUNC(memmove)(void* s1, const void* s2, size_t n);
|
|
||||||
|
|
||||||
/* Copy the character array s2 (including terminating '\0' byte) into the
|
|
||||||
character array s1.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strcpy)(char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Copy a maximum of n characters from the character array s2 into the character
|
|
||||||
array s1. If s2 is shorter than n characters, '\0' bytes will be appended to
|
|
||||||
the copy in s1 until n characters have been written. If s2 is longer than n
|
|
||||||
characters, NO terminating '\0' will be written to s1. If the arrays overlap,
|
|
||||||
behaviour is undefined.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strncpy)(char* s1, const char* s2, size_t n);
|
|
||||||
|
|
||||||
/* Concatenation functions */
|
|
||||||
|
|
||||||
/* Append the contents of the character array s2 (including terminating '\0') to
|
|
||||||
the character array s1 (first character of s2 overwriting the '\0' of s1). If
|
|
||||||
the arrays overlap, behaviour is undefined.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strcat)(char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Append a maximum of n characters from the character array s2 to the character
|
|
||||||
array s1 (first character of s2 overwriting the '\0' of s1). A terminating
|
|
||||||
'\0' is ALWAYS appended, even if the full n characters have already been
|
|
||||||
written. If the arrays overlap, behaviour is undefined.
|
|
||||||
Returns the value of s1.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strncat)(char* s1, const char* s2, size_t n);
|
|
||||||
|
|
||||||
/* Comparison functions */
|
|
||||||
|
|
||||||
/* Compare the first n characters of the memory areas pointed to by s1 and s2.
|
|
||||||
Returns 0 if s1 == s2, a negative number if s1 < s2, and a positive number if
|
|
||||||
s1 > s2.
|
|
||||||
*/
|
|
||||||
extern int _FUNC(memcmp)(const void * s1, const void* s2, size_t n);
|
|
||||||
|
|
||||||
/* Compare the character arrays s1 and s2.
|
|
||||||
Returns 0 if s1 == s2, a negative number if s1 < s2, and a positive number if
|
|
||||||
s1 > s2.
|
|
||||||
*/
|
|
||||||
extern int _FUNC(strcmp)(const char * s1, const char* s2);
|
|
||||||
|
|
||||||
/* Compare the character arrays s1 and s2, interpreted as specified by the
|
|
||||||
LC_COLLATE category of the current locale.
|
|
||||||
Returns 0 if s1 == s2, a negative number if s1 < s2, and a positive number if
|
|
||||||
s1 > s2.
|
|
||||||
TODO: Currently a dummy wrapper for strcmp() as PDCLib does not yet support
|
|
||||||
locales.
|
|
||||||
*/
|
|
||||||
extern int _FUNC(strcoll)(const char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Compare no more than the first n characters of the character arrays s1 and
|
|
||||||
s2.
|
|
||||||
Returns 0 if s1 == s2, a negative number if s1 < s2, and a positive number if
|
|
||||||
s1 > s2.
|
|
||||||
*/
|
|
||||||
extern int _FUNC(strncmp)(const char* s1, const char* s2, size_t n);
|
|
||||||
|
|
||||||
/* Transform the character array s2 as appropriate for the LC_COLLATE setting of
|
|
||||||
the current locale. If length of resulting string is less than n, store it in
|
|
||||||
the character array pointed to by s1. Return the length of the resulting
|
|
||||||
string.
|
|
||||||
*/
|
|
||||||
extern size_t _FUNC(strxfrm)(char* s1, const char* s2, size_t n);
|
|
||||||
|
|
||||||
/* Search functions */
|
|
||||||
|
|
||||||
/* Search the first n characters in the memory area pointed to by s for the
|
|
||||||
character c (interpreted as unsigned char).
|
|
||||||
Returns a pointer to the first instance found, or NULL.
|
|
||||||
*/
|
|
||||||
extern void* _FUNC(memchr)(const void* s, int c, size_t n);
|
|
||||||
|
|
||||||
/* Search the character array s (including terminating '\0') for the character c
|
|
||||||
(interpreted as char).
|
|
||||||
Returns a pointer to the first instance found, or NULL.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strchr)(const char* s, int c);
|
|
||||||
|
|
||||||
/* Determine the length of the initial substring of character array s1 which
|
|
||||||
consists only of characters not from the character array s2.
|
|
||||||
Returns the length of that substring.
|
|
||||||
*/
|
|
||||||
extern size_t _FUNC(strcspn)(const char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Search the character array s1 for any character from the character array s2.
|
|
||||||
Returns a pointer to the first occurrence, or NULL.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strpbrk)(const char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Search the character array s (including terminating '\0') for the character c
|
|
||||||
(interpreted as char).
|
|
||||||
Returns a pointer to the last instance found, or NULL.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strrchr)(const char * s, int c );
|
|
||||||
|
|
||||||
/* Determine the length of the initial substring of character array s1 which
|
|
||||||
consists only of characters from the character array s2.
|
|
||||||
Returns the length of that substring.
|
|
||||||
*/
|
|
||||||
extern size_t _FUNC(strspn)(const char * s1, const char * s2);
|
|
||||||
|
|
||||||
/* Search the character array s1 for the substring in character array s2.
|
|
||||||
Returns a pointer to that sbstring, or NULL. If s2 is of length zero,
|
|
||||||
returns s1.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strstr)(const char * s1, const char * s2);
|
|
||||||
|
|
||||||
/* In a series of subsequent calls, parse a C string into tokens.
|
|
||||||
On the first call to strtok(), the first argument is a pointer to the to-be-
|
|
||||||
parsed C string. On subsequent calls, the first argument is NULL unless you
|
|
||||||
want to start parsing a new string. s2 holds an array of separator characters
|
|
||||||
which can differ from call to call. Leading separators are skipped, the first
|
|
||||||
trailing separator overwritten with '\0'.
|
|
||||||
Returns a pointer to the next token.
|
|
||||||
WARNING: This function uses static storage, and as such is not reentrant.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strtok)(char* s1, const char* s2);
|
|
||||||
|
|
||||||
/* Miscellaneous functions */
|
|
||||||
|
|
||||||
/* Write the character c (interpreted as unsigned char) to the first n
|
|
||||||
characters of the memory area pointed to by s.
|
|
||||||
Returns s.
|
|
||||||
*/
|
|
||||||
extern void* _FUNC(memset)(void* s, int c, size_t n);
|
|
||||||
|
|
||||||
/* Map an error number to a (locale-specific) error message string. Error
|
|
||||||
numbers are typically errno values, but any number is mapped to a message.
|
|
||||||
TODO: PDCLib does not yet support locales.
|
|
||||||
*/
|
|
||||||
extern char* _FUNC(strerror)(int errnum);
|
|
||||||
|
|
||||||
/* Returns the length of the string s (excluding terminating '\0').*/
|
|
||||||
extern size_t _FUNC(strlen)(const char * s);
|
|
||||||
|
|
||||||
/* The function reverses the sequence of characters in the string pointed to by str. */
|
|
||||||
extern char* _FUNC(strrev)(char *str);
|
|
||||||
|
|
||||||
/* The strdup function executes the function pointed to by the str argument. */
|
|
||||||
extern char* _FUNC(strdup)(const char *str);
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,34 +0,0 @@
|
|||||||
/* Copyright (C) 2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
||||||
|
|
||||||
#ifndef _DIRENT_H_
|
|
||||||
#define _DIRENT_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
#include <limits.h>
|
|
||||||
|
|
||||||
#define IS_FOLDER 16
|
|
||||||
#define IS_FILE 0
|
|
||||||
|
|
||||||
typedef unsigned ino_t;
|
|
||||||
|
|
||||||
struct dirent{
|
|
||||||
ino_t d_ino; //File serial number.
|
|
||||||
char d_name[PATH_MAX]; // Name of entry.
|
|
||||||
unsigned d_type;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
struct dirent* objs;
|
|
||||||
ino_t pos;
|
|
||||||
ino_t num_objs;
|
|
||||||
}DIR;
|
|
||||||
|
|
||||||
|
|
||||||
int _FUNC(closedir)(DIR *dir);
|
|
||||||
DIR* _FUNC(opendir)(const char *path);
|
|
||||||
struct dirent* _FUNC(readdir)(DIR *);
|
|
||||||
void _FUNC(rewinddir)(DIR *dir);
|
|
||||||
void _FUNC(seekdir)(DIR *dir, unsigned pos);
|
|
||||||
unsigned _FUNC(telldir)(DIR *dir);
|
|
||||||
|
|
||||||
#endif // _DIRENT_H_
|
|
@ -1,244 +0,0 @@
|
|||||||
/* Copyright (C) 2019-2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
||||||
|
|
||||||
#ifndef _SOCKET_H_
|
|
||||||
#define _SOCKET_H_
|
|
||||||
|
|
||||||
#include <stddef.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
#include <errno.h>
|
|
||||||
|
|
||||||
// Socket Types
|
|
||||||
#define SOCK_STREAM 1
|
|
||||||
#define SOCK_DGRAM 2
|
|
||||||
#define SOCK_RAW 3
|
|
||||||
|
|
||||||
// IP protocols
|
|
||||||
#define IPPROTO_IP 0
|
|
||||||
#define IPPROTO_ICMP 1
|
|
||||||
#define IPPROTO_TCP 6
|
|
||||||
#define IPPROTO_UDP 17
|
|
||||||
#define IPPROTO_RAW 255
|
|
||||||
|
|
||||||
// IP options
|
|
||||||
#define IP_TTL 2
|
|
||||||
|
|
||||||
// Address families
|
|
||||||
#define AF_UNSPEC 0
|
|
||||||
#define AF_LOCAL 1
|
|
||||||
#define AF_INET 2 // Default INET=IPv4
|
|
||||||
#define AF_INET4 2 // IPv4
|
|
||||||
#define AF_INET6 10 // IPv6
|
|
||||||
|
|
||||||
#define PF_UNSPEC AF_UNSPEC
|
|
||||||
#define PF_LOCAL AF_LOCAL
|
|
||||||
#define PF_INET4 AF_INET4
|
|
||||||
#define PF_INET6 AF_INET6
|
|
||||||
|
|
||||||
// internal definition
|
|
||||||
#define AI_SUPPORTED 0x40F
|
|
||||||
|
|
||||||
// for system function 76
|
|
||||||
#define API_ETH (0<<16)
|
|
||||||
#define API_IPv4 (1<<16)
|
|
||||||
#define API_ICMP (2<<16)
|
|
||||||
#define API_UDP (3<<16)
|
|
||||||
#define API_TCP (4<<16)
|
|
||||||
#define API_ARP (5<<16)
|
|
||||||
#define API_PPPOE (6<<16)
|
|
||||||
|
|
||||||
// Socket flags for user calls
|
|
||||||
#define MSG_NOFLAG 0
|
|
||||||
#define MSG_PEEK 0x02
|
|
||||||
#define MSG_DONTWAIT 0x40
|
|
||||||
|
|
||||||
// Socket levels
|
|
||||||
#define SOL_SOCKET 0xffff
|
|
||||||
|
|
||||||
//Socket options
|
|
||||||
#define SO_BINDTODEVICE (1<<9)
|
|
||||||
#define SO_NONBLOCK (1<<31)
|
|
||||||
|
|
||||||
#define PORT(X) (X<<8)
|
|
||||||
|
|
||||||
#pragma pack(push,1)
|
|
||||||
struct sockaddr{
|
|
||||||
unsigned short sin_family;
|
|
||||||
unsigned short sin_port;
|
|
||||||
unsigned int sin_addr;
|
|
||||||
unsigned long long sin_zero;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct{
|
|
||||||
unsigned int level;
|
|
||||||
unsigned int optionname;
|
|
||||||
unsigned int optlenght;
|
|
||||||
unsigned char options;
|
|
||||||
}optstruct;
|
|
||||||
#pragma pack(pop)
|
|
||||||
|
|
||||||
static inline
|
|
||||||
void _conv_socket_err(){
|
|
||||||
switch(errno){
|
|
||||||
case 1: errno = ENOBUFS; break;
|
|
||||||
case 2: errno = EINPROGRESS; break;
|
|
||||||
case 4: errno = EOPNOTSUPP; break;
|
|
||||||
case 6: errno = EWOULDBLOCK; break;
|
|
||||||
case 9: errno = ENOTCONN; break;
|
|
||||||
case 10: errno = EALREADY; break;
|
|
||||||
case 11: errno = EINVAL; break;
|
|
||||||
case 12: errno = EMSGSIZE; break;
|
|
||||||
case 18: errno = ENOMEM; break;
|
|
||||||
case 20: errno = EADDRINUSE; break;
|
|
||||||
case 61: errno = ECONNREFUSED; break;
|
|
||||||
case 52: errno = ECONNRESET; break;
|
|
||||||
case 56: errno = EISCONN; break;
|
|
||||||
case 60: errno = ETIMEDOUT; break;
|
|
||||||
case 54: errno = ECONNABORTED; break;
|
|
||||||
default: errno = 0; break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int socket(int domain, int type, int protocol)
|
|
||||||
{
|
|
||||||
int socket;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(socket)
|
|
||||||
:"a"(75), "b"(0), "c"(domain), "d"(type), "S"(protocol)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return socket;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int close(int socket)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(1), "c"(socket)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int bind(int socket, const struct sockaddr *addres, int addres_len)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(2), "c"(socket), "d"(addres), "S"(addres_len)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int listen(int socket, int backlog)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(3), "c"(socket), "d"(backlog)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int connect(int socket, const struct sockaddr* address, int socket_len)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(4), "c"(socket), "d"(address), "S"(socket_len)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline int
|
|
||||||
accept(int socket, const struct sockaddr *address, int address_len)
|
|
||||||
{
|
|
||||||
int new_socket;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(new_socket)
|
|
||||||
:"a"(75), "b"(5), "c"(socket), "d"(address), "S"(address_len)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return new_socket;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int send(int socket, const void *message, size_t msg_len, int flag)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(6), "c"(socket), "d"(message), "S"(msg_len), "D"(flag)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int recv(int socket, void *buffer, size_t buff_len, int flag)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(7), "c"(socket), "d"(buffer), "S"(buff_len), "D"(flag)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int setsockopt(int socket,const optstruct* opt)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(8), "c"(socket),"d"(opt)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int getsockopt(int socket, optstruct* opt)
|
|
||||||
{
|
|
||||||
int status;
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(errno), "=a"(status)
|
|
||||||
:"a"(75), "b"(9), "c"(socket),"d"(opt)
|
|
||||||
);
|
|
||||||
_conv_socket_err();
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline
|
|
||||||
int socketpair(int *socket1, int *socket2)
|
|
||||||
{
|
|
||||||
asm_inline(
|
|
||||||
"int $0x40"
|
|
||||||
:"=b"(*socket2), "=a"(*socket1)
|
|
||||||
:"a"(75), "b"(10)
|
|
||||||
);
|
|
||||||
errno=*socket2;
|
|
||||||
_conv_socket_err();
|
|
||||||
return *socket1;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif //_SOCKET_H_
|
|
@ -1,30 +0,0 @@
|
|||||||
#ifndef _TIME_H_
|
|
||||||
#define _TIME_H_
|
|
||||||
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
typedef unsigned long int clock_t;
|
|
||||||
typedef unsigned long int time_t;
|
|
||||||
#define clock() _ksys_get_clock()
|
|
||||||
#define CLOCKS_PER_SEC 100
|
|
||||||
|
|
||||||
struct tm {
|
|
||||||
int tm_sec; /* seconds after the minute 0-61*/
|
|
||||||
int tm_min; /* minutes after the hour 0-59 */
|
|
||||||
int tm_hour; /* hours since midnight 0-23 */
|
|
||||||
int tm_mday; /* day of the month 1-31 */
|
|
||||||
int tm_mon; /* months since January 0-11 */
|
|
||||||
int tm_year; /* years since 1900 */
|
|
||||||
int tm_wday; /* days since Sunday 0-6 */
|
|
||||||
int tm_yday; /* days since January 1 0-365 */
|
|
||||||
int tm_isdst; /* Daylight Saving Time flag */
|
|
||||||
};
|
|
||||||
|
|
||||||
extern time_t _FUNC(mktime)(struct tm * timeptr);
|
|
||||||
extern time_t _FUNC(time)(time_t* timer);
|
|
||||||
extern struct tm * _FUNC(localtime)(const time_t * timer); /* non-standard! ignore parameter and return just time now, not generate tm_isdst, tm_yday, tm_wday == -1 */
|
|
||||||
extern double _FUNC(difftime)(time_t end, time_t beginning);
|
|
||||||
|
|
||||||
extern struct tm buffertime;
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,22 +0,0 @@
|
|||||||
#include "libc/asm.h"
|
|
||||||
|
|
||||||
.text
|
|
||||||
LC0:
|
|
||||||
.double 0d1.00000000000000000000e+00
|
|
||||||
|
|
||||||
MK_C_SYM(acos)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fld1
|
|
||||||
fsubp %st(0),%st(1)
|
|
||||||
fsqrt
|
|
||||||
|
|
||||||
fldl 4(%esp)
|
|
||||||
fld1
|
|
||||||
faddp %st(0),%st(1)
|
|
||||||
fsqrt
|
|
||||||
|
|
||||||
fpatan
|
|
||||||
|
|
||||||
fld %st(0)
|
|
||||||
faddp
|
|
||||||
ret
|
|
@ -1,8 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double
|
|
||||||
acosh(double x)
|
|
||||||
{
|
|
||||||
return log(x + sqrt(x*x - 1));
|
|
||||||
}
|
|
@ -1,14 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(asin)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fld %st(0)
|
|
||||||
fmulp
|
|
||||||
fld1
|
|
||||||
fsubp
|
|
||||||
fsqrt
|
|
||||||
fldl 4(%esp)
|
|
||||||
fxch %st(1)
|
|
||||||
fpatan
|
|
||||||
ret
|
|
||||||
|
|
@ -1,9 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double
|
|
||||||
asinh(double x)
|
|
||||||
{
|
|
||||||
return x>0 ? log(x + sqrt(x*x + 1)) : -log(sqrt(x*x+1)-x);
|
|
||||||
}
|
|
||||||
|
|
@ -1,8 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(atan)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fld1
|
|
||||||
fpatan
|
|
||||||
ret
|
|
||||||
|
|
@ -1,35 +0,0 @@
|
|||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
|
|
||||||
.data
|
|
||||||
.align 2
|
|
||||||
nan:
|
|
||||||
.long 0xffffffff
|
|
||||||
.byte 0xff
|
|
||||||
.byte 0xff
|
|
||||||
.byte 0xff
|
|
||||||
.byte 0x7f
|
|
||||||
|
|
||||||
.text
|
|
||||||
MK_C_SYM(atan2)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fldl 12(%esp)
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
sahf
|
|
||||||
jne doit
|
|
||||||
fxch %st(1)
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
sahf
|
|
||||||
je isanan
|
|
||||||
fxch %st(1)
|
|
||||||
doit:
|
|
||||||
fpatan
|
|
||||||
ret
|
|
||||||
isanan:
|
|
||||||
movl $1,C_SYM(errno)
|
|
||||||
fstp %st(0)
|
|
||||||
fstp %st(0)
|
|
||||||
fldl nan
|
|
||||||
ret
|
|
@ -1,8 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double
|
|
||||||
atanh(double x)
|
|
||||||
{
|
|
||||||
return log((1+x)/(1-x)) / 2.0;
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(ceil)
|
|
||||||
pushl %ebp
|
|
||||||
movl %esp,%ebp
|
|
||||||
subl $8,%esp
|
|
||||||
|
|
||||||
fstcw -4(%ebp)
|
|
||||||
fwait
|
|
||||||
movw -4(%ebp),%ax
|
|
||||||
andw $0xf3ff,%ax
|
|
||||||
orw $0x0800,%ax
|
|
||||||
movw %ax,-2(%ebp)
|
|
||||||
fldcw -2(%ebp)
|
|
||||||
|
|
||||||
fldl 8(%ebp)
|
|
||||||
frndint
|
|
||||||
|
|
||||||
fldcw -4(%ebp)
|
|
||||||
|
|
||||||
movl %ebp,%esp
|
|
||||||
popl %ebp
|
|
||||||
ret
|
|
||||||
|
|
@ -1,15 +0,0 @@
|
|||||||
#include "libc/asm.h"
|
|
||||||
L0:
|
|
||||||
.quad 0xffffffffffffffff
|
|
||||||
|
|
||||||
MK_C_SYM(cos)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fcos
|
|
||||||
fstsw
|
|
||||||
sahf
|
|
||||||
jnp L1
|
|
||||||
fstp %st(0)
|
|
||||||
fldl L0
|
|
||||||
L1:
|
|
||||||
ret
|
|
||||||
|
|
@ -1,8 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double cosh(double x)
|
|
||||||
{
|
|
||||||
const double ebig = exp(fabs(x));
|
|
||||||
return (ebig + 1.0/ebig) / 2.0;
|
|
||||||
}
|
|
@ -1,31 +0,0 @@
|
|||||||
#include "libc/asm.h"
|
|
||||||
.data
|
|
||||||
LCW1:
|
|
||||||
.word 0
|
|
||||||
LCW2:
|
|
||||||
.word 0
|
|
||||||
LC0:
|
|
||||||
.double 0d1.0e+00
|
|
||||||
|
|
||||||
.text
|
|
||||||
|
|
||||||
MK_C_SYM(exp)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fldl2e
|
|
||||||
fmulp
|
|
||||||
fstcw LCW1
|
|
||||||
fstcw LCW2
|
|
||||||
fwait
|
|
||||||
andw $0xf3ff,LCW2
|
|
||||||
orw $0x0400,LCW2
|
|
||||||
fldcw LCW2
|
|
||||||
fldl %st(0)
|
|
||||||
frndint
|
|
||||||
fldcw LCW1
|
|
||||||
fxch %st(1)
|
|
||||||
fsub %st(1),%st
|
|
||||||
f2xm1
|
|
||||||
faddl LC0
|
|
||||||
fscale
|
|
||||||
fstp %st(1)
|
|
||||||
ret
|
|
@ -1,6 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(fabs)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fabs
|
|
||||||
ret
|
|
@ -1,24 +0,0 @@
|
|||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(floor)
|
|
||||||
pushl %ebp
|
|
||||||
movl %esp,%ebp
|
|
||||||
subl $8,%esp
|
|
||||||
|
|
||||||
fstcw -4(%ebp)
|
|
||||||
fwait
|
|
||||||
movw -4(%ebp),%ax
|
|
||||||
andw $0xf3ff,%ax
|
|
||||||
orw $0x0400,%ax
|
|
||||||
movw %ax,-2(%ebp)
|
|
||||||
fldcw -2(%ebp)
|
|
||||||
|
|
||||||
fldl 8(%ebp)
|
|
||||||
frndint
|
|
||||||
|
|
||||||
fldcw -4(%ebp)
|
|
||||||
|
|
||||||
movl %ebp,%esp
|
|
||||||
popl %ebp
|
|
||||||
ret
|
|
||||||
|
|
@ -1,29 +0,0 @@
|
|||||||
#include "libc/asm.h"
|
|
||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
.data
|
|
||||||
LCW1:
|
|
||||||
.word 0
|
|
||||||
.align 4
|
|
||||||
|
|
||||||
.text
|
|
||||||
|
|
||||||
MK_C_SYM(fmod)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fldl 12(%esp)
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
fxch %st(1)
|
|
||||||
sahf
|
|
||||||
jnz next
|
|
||||||
fstpl %st(0)
|
|
||||||
jmp out
|
|
||||||
next:
|
|
||||||
fprem
|
|
||||||
fnstsw %ax
|
|
||||||
sahf
|
|
||||||
jpe next
|
|
||||||
fstpl %st(1)
|
|
||||||
out:
|
|
||||||
ret
|
|
||||||
|
|
||||||
|
|
@ -1,26 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double
|
|
||||||
frexp(double x, int *exptr)
|
|
||||||
{
|
|
||||||
union {
|
|
||||||
double d;
|
|
||||||
unsigned char c[8];
|
|
||||||
} u;
|
|
||||||
|
|
||||||
u.d = x;
|
|
||||||
/*
|
|
||||||
* The format of the number is:
|
|
||||||
* Sign, 12 exponent bits, 51 mantissa bits
|
|
||||||
* The exponent is 1023 biased and there is an implicit zero.
|
|
||||||
* We get the exponent from the upper bits and set the exponent
|
|
||||||
* to 0x3fe (1022).
|
|
||||||
*/
|
|
||||||
*exptr = (int)(((u.c[7] & 0x7f) << 4) | (u.c[6] >> 4)) - 1022;
|
|
||||||
u.c[7] &= 0x80;
|
|
||||||
u.c[7] |= 0x3f;
|
|
||||||
u.c[6] &= 0x0f;
|
|
||||||
u.c[6] |= 0xe0;
|
|
||||||
return u.d;
|
|
||||||
}
|
|
@ -1,100 +0,0 @@
|
|||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
/*
|
|
||||||
* hypot() function for DJGPP.
|
|
||||||
*
|
|
||||||
* hypot() computes sqrt(x^2 + y^2). The problem with the obvious
|
|
||||||
* naive implementation is that it might fail for very large or
|
|
||||||
* very small arguments. For instance, for large x or y the result
|
|
||||||
* might overflow even if the value of the function should not,
|
|
||||||
* because squaring a large number might trigger an overflow. For
|
|
||||||
* very small numbers, their square might underflow and will be
|
|
||||||
* silently replaced by zero; this won't cause an exception, but might
|
|
||||||
* have an adverse effect on the accuracy of the result.
|
|
||||||
*
|
|
||||||
* This implementation tries to avoid the above pitfals, without
|
|
||||||
* inflicting too much of a performance hit.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
/// #include <float.h>
|
|
||||||
#include <math.h>
|
|
||||||
#include <errno.h>
|
|
||||||
|
|
||||||
/* Approximate square roots of DBL_MAX and DBL_MIN. Numbers
|
|
||||||
between these two shouldn't neither overflow nor underflow
|
|
||||||
when squared. */
|
|
||||||
#define __SQRT_DBL_MAX 1.3e+154
|
|
||||||
#define __SQRT_DBL_MIN 2.3e-162
|
|
||||||
|
|
||||||
double
|
|
||||||
hypot(double x, double y)
|
|
||||||
{
|
|
||||||
double abig = fabs(x), asmall = fabs(y);
|
|
||||||
double ratio;
|
|
||||||
|
|
||||||
/* Make abig = max(|x|, |y|), asmall = min(|x|, |y|). */
|
|
||||||
if (abig < asmall)
|
|
||||||
{
|
|
||||||
double temp = abig;
|
|
||||||
|
|
||||||
abig = asmall;
|
|
||||||
asmall = temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Trivial case. */
|
|
||||||
if (asmall == 0.)
|
|
||||||
return abig;
|
|
||||||
|
|
||||||
/* Scale the numbers as much as possible by using its ratio.
|
|
||||||
For example, if both ABIG and ASMALL are VERY small, then
|
|
||||||
X^2 + Y^2 might be VERY inaccurate due to loss of
|
|
||||||
significant digits. Dividing ASMALL by ABIG scales them
|
|
||||||
to a certain degree, so that accuracy is better. */
|
|
||||||
|
|
||||||
if ((ratio = asmall / abig) > __SQRT_DBL_MIN && abig < __SQRT_DBL_MAX)
|
|
||||||
return abig * sqrt(1.0 + ratio*ratio);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Slower but safer algorithm due to Moler and Morrison. Never
|
|
||||||
produces any intermediate result greater than roughly the
|
|
||||||
larger of X and Y. Should converge to machine-precision
|
|
||||||
accuracy in 3 iterations. */
|
|
||||||
|
|
||||||
double r = ratio*ratio, t, s, p = abig, q = asmall;
|
|
||||||
|
|
||||||
do {
|
|
||||||
t = 4. + r;
|
|
||||||
if (t == 4.)
|
|
||||||
break;
|
|
||||||
s = r / t;
|
|
||||||
p += 2. * s * p;
|
|
||||||
q *= s;
|
|
||||||
r = (q / p) * (q / p);
|
|
||||||
} while (1);
|
|
||||||
|
|
||||||
return p;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef TEST
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int
|
|
||||||
main(void)
|
|
||||||
{
|
|
||||||
printf("hypot(3, 4) =\t\t\t %25.17e\n", hypot(3., 4.));
|
|
||||||
printf("hypot(3*10^150, 4*10^150) =\t %25.17g\n", hypot(3.e+150, 4.e+150));
|
|
||||||
printf("hypot(3*10^306, 4*10^306) =\t %25.17g\n", hypot(3.e+306, 4.e+306));
|
|
||||||
printf("hypot(3*10^-320, 4*10^-320) =\t %25.17g\n",
|
|
||||||
hypot(3.e-320, 4.e-320));
|
|
||||||
printf("hypot(0.7*DBL_MAX, 0.7*DBL_MAX) =%25.17g\n",
|
|
||||||
hypot(0.7*DBL_MAX, 0.7*DBL_MAX));
|
|
||||||
printf("hypot(DBL_MAX, 1.0) =\t\t %25.17g\n", hypot(DBL_MAX, 1.0));
|
|
||||||
printf("hypot(1.0, DBL_MAX) =\t\t %25.17g\n", hypot(1.0, DBL_MAX));
|
|
||||||
printf("hypot(0.0, DBL_MAX) =\t\t %25.17g\n", hypot(0.0, DBL_MAX));
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,32 +0,0 @@
|
|||||||
/* Copyright (C) 1996 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double
|
|
||||||
ldexp(double v, int e)
|
|
||||||
{
|
|
||||||
double two = 2.0;
|
|
||||||
|
|
||||||
if (e < 0)
|
|
||||||
{
|
|
||||||
e = -e; /* This just might overflow on two-complement machines. */
|
|
||||||
if (e < 0) return 0.0;
|
|
||||||
while (e > 0)
|
|
||||||
{
|
|
||||||
if (e & 1) v /= two;
|
|
||||||
two *= two;
|
|
||||||
e >>= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (e > 0)
|
|
||||||
{
|
|
||||||
while (e > 0)
|
|
||||||
{
|
|
||||||
if (e & 1) v *= two;
|
|
||||||
two *= two;
|
|
||||||
e >>= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
|
|
@ -1,9 +0,0 @@
|
|||||||
#ifndef __LIBC_ASM_H
|
|
||||||
#define __LIBC_ASM_H
|
|
||||||
|
|
||||||
#define C_SYM(x) x
|
|
||||||
// #define C_SYM(x) _##x
|
|
||||||
|
|
||||||
#define MK_C_SYM(x) .global C_SYM(x); C_SYM(x):
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,7 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(log)
|
|
||||||
fldln2
|
|
||||||
fldl 4(%esp)
|
|
||||||
fyl2x
|
|
||||||
ret
|
|
@ -1,33 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
.text
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(modf)
|
|
||||||
pushl %ebp
|
|
||||||
movl %esp,%ebp
|
|
||||||
subl $16,%esp
|
|
||||||
pushl %ebx
|
|
||||||
fnstcw -4(%ebp)
|
|
||||||
fwait
|
|
||||||
movw -4(%ebp),%ax
|
|
||||||
orw $0x0c3f,%ax
|
|
||||||
movw %ax,-8(%ebp)
|
|
||||||
fldcw -8(%ebp)
|
|
||||||
fwait
|
|
||||||
fldl 8(%ebp)
|
|
||||||
frndint
|
|
||||||
fstpl -16(%ebp)
|
|
||||||
fwait
|
|
||||||
movl -16(%ebp),%edx
|
|
||||||
movl -12(%ebp),%ecx
|
|
||||||
movl 16(%ebp),%ebx
|
|
||||||
movl %edx,(%ebx)
|
|
||||||
movl %ecx,4(%ebx)
|
|
||||||
fldl 8(%ebp)
|
|
||||||
fsubl -16(%ebp)
|
|
||||||
leal -20(%ebp),%esp
|
|
||||||
fclex
|
|
||||||
fldcw -4(%ebp)
|
|
||||||
fwait
|
|
||||||
popl %ebx
|
|
||||||
leave
|
|
||||||
ret
|
|
@ -1,22 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(__modfl)
|
|
||||||
MK_C_SYM(modfl)
|
|
||||||
pushl %ebp
|
|
||||||
movl %esp,%ebp
|
|
||||||
subl $4,%esp
|
|
||||||
fldt 8(%ebp)
|
|
||||||
movl 20(%ebp),%eax
|
|
||||||
fnstcw -2(%ebp)
|
|
||||||
movw -2(%ebp),%dx
|
|
||||||
orb $0x0c,%dh
|
|
||||||
movw %dx,-4(%ebp)
|
|
||||||
fldcw -4(%ebp)
|
|
||||||
fld %st(0)
|
|
||||||
frndint
|
|
||||||
fldcw -2(%ebp)
|
|
||||||
fld %st(0)
|
|
||||||
fstpt (%eax)
|
|
||||||
fsubrp %st,%st(1)
|
|
||||||
leave
|
|
||||||
ret
|
|
@ -1,86 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
.data
|
|
||||||
yint:
|
|
||||||
.word 0,0
|
|
||||||
LCW1:
|
|
||||||
.word 0
|
|
||||||
LCW2:
|
|
||||||
.word 0
|
|
||||||
|
|
||||||
.text
|
|
||||||
LC0:
|
|
||||||
.double 0d1.0e+00
|
|
||||||
|
|
||||||
frac:
|
|
||||||
fstcw LCW1
|
|
||||||
fstcw LCW2
|
|
||||||
fwait
|
|
||||||
andw $0xf3ff,LCW2
|
|
||||||
orw $0x0400,LCW2
|
|
||||||
fldcw LCW2
|
|
||||||
fldl %st(0)
|
|
||||||
frndint
|
|
||||||
fldcw LCW1
|
|
||||||
fxch %st(1)
|
|
||||||
fsub %st(1),%st
|
|
||||||
ret
|
|
||||||
|
|
||||||
Lpow2:
|
|
||||||
call frac
|
|
||||||
f2xm1
|
|
||||||
faddl LC0
|
|
||||||
fscale
|
|
||||||
fstp %st(1)
|
|
||||||
ret
|
|
||||||
|
|
||||||
MK_C_SYM(pow)
|
|
||||||
fldl 12(%esp)
|
|
||||||
fldl 4(%esp)
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
sahf
|
|
||||||
jbe xltez
|
|
||||||
fyl2x
|
|
||||||
jmp Lpow2
|
|
||||||
xltez:
|
|
||||||
jb xltz
|
|
||||||
fstp %st(0)
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
sahf
|
|
||||||
ja ygtz
|
|
||||||
jb error
|
|
||||||
fstp %st(0)
|
|
||||||
fld1
|
|
||||||
fchs
|
|
||||||
error:
|
|
||||||
fsqrt
|
|
||||||
ret
|
|
||||||
ygtz:
|
|
||||||
fstp %st(0)
|
|
||||||
fldz
|
|
||||||
ret
|
|
||||||
xltz:
|
|
||||||
fabs
|
|
||||||
fxch %st(1)
|
|
||||||
call frac
|
|
||||||
ftst
|
|
||||||
fnstsw %ax
|
|
||||||
fstp %st(0)
|
|
||||||
sahf
|
|
||||||
je yisint
|
|
||||||
fstp %st(0)
|
|
||||||
fchs
|
|
||||||
jmp error
|
|
||||||
yisint:
|
|
||||||
fistl yint
|
|
||||||
fxch %st(1)
|
|
||||||
fyl2x
|
|
||||||
call Lpow2
|
|
||||||
andl $1,yint
|
|
||||||
jz yeven
|
|
||||||
fchs
|
|
||||||
yeven:
|
|
||||||
ret
|
|
||||||
|
|
@ -1,33 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
.data
|
|
||||||
LCW1:
|
|
||||||
.word 0
|
|
||||||
LCW2:
|
|
||||||
.word 0
|
|
||||||
LC0:
|
|
||||||
.double 0d1.0e+00
|
|
||||||
|
|
||||||
.text
|
|
||||||
|
|
||||||
MK_C_SYM(__pow10)
|
|
||||||
MK_C_SYM(pow10)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fldl2t
|
|
||||||
fmulp
|
|
||||||
fstcw LCW1
|
|
||||||
fstcw LCW2
|
|
||||||
fwait
|
|
||||||
andw $0xf3ff,LCW2
|
|
||||||
orw $0x0400,LCW2
|
|
||||||
fldcw LCW2
|
|
||||||
fldl %st(0)
|
|
||||||
frndint
|
|
||||||
fldcw LCW1
|
|
||||||
fxch %st(1)
|
|
||||||
fsub %st(1),%st
|
|
||||||
f2xm1
|
|
||||||
faddl LC0
|
|
||||||
fscale
|
|
||||||
fstp %st(1)
|
|
||||||
ret
|
|
@ -1,31 +0,0 @@
|
|||||||
/* Copyright (C) 1995 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
.data
|
|
||||||
LCW1:
|
|
||||||
.word 0
|
|
||||||
LCW2:
|
|
||||||
.word 0
|
|
||||||
LC0:
|
|
||||||
.double 0d1.0e+00
|
|
||||||
|
|
||||||
.text
|
|
||||||
|
|
||||||
MK_C_SYM(__pow2)
|
|
||||||
MK_C_SYM(pow2)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fstcw LCW1
|
|
||||||
fstcw LCW2
|
|
||||||
fwait
|
|
||||||
andw $0xf3ff,LCW2
|
|
||||||
orw $0x0400,LCW2
|
|
||||||
fldcw LCW2
|
|
||||||
fldl %st(0)
|
|
||||||
frndint
|
|
||||||
fldcw LCW1
|
|
||||||
fxch %st(1)
|
|
||||||
fsub %st(1),%st
|
|
||||||
f2xm1
|
|
||||||
faddl LC0
|
|
||||||
fscale
|
|
||||||
fstp %st(1)
|
|
||||||
ret
|
|
@ -1,16 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
L0:
|
|
||||||
.quad 0xffffffffffffffff
|
|
||||||
|
|
||||||
MK_C_SYM(sin)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fsin
|
|
||||||
fstsw
|
|
||||||
sahf
|
|
||||||
jnp L1
|
|
||||||
fstp %st(0)
|
|
||||||
fldl L0
|
|
||||||
L1:
|
|
||||||
ret
|
|
||||||
|
|
@ -1,16 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double sinh(double x)
|
|
||||||
{
|
|
||||||
if(x >= 0.0)
|
|
||||||
{
|
|
||||||
const double epos = exp(x);
|
|
||||||
return (epos - 1.0/epos) / 2.0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
const double eneg = exp(-x);
|
|
||||||
return (1.0/eneg - eneg) / 2.0;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
MK_C_SYM(sqrt)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fsqrt
|
|
||||||
ret
|
|
@ -1,16 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include "libc/asm.h"
|
|
||||||
L0:
|
|
||||||
.quad 0xffffffffffffffff
|
|
||||||
|
|
||||||
MK_C_SYM(tan)
|
|
||||||
fldl 4(%esp)
|
|
||||||
fptan
|
|
||||||
fstsw
|
|
||||||
fstp %st(0)
|
|
||||||
sahf
|
|
||||||
jnp L1
|
|
||||||
/* fstp %st(0) - if exception, there is nothing on the stack */
|
|
||||||
fldl L0
|
|
||||||
L1:
|
|
||||||
ret
|
|
@ -1,17 +0,0 @@
|
|||||||
/* Copyright (C) 1994 DJ Delorie, see COPYING.DJ for details */
|
|
||||||
#include <math.h>
|
|
||||||
|
|
||||||
double tanh(double x)
|
|
||||||
{
|
|
||||||
if (x > 50)
|
|
||||||
return 1;
|
|
||||||
else if (x < -50)
|
|
||||||
return -1;
|
|
||||||
else
|
|
||||||
{
|
|
||||||
const double ebig = exp(x);
|
|
||||||
const double esmall = 1.0/ebig;
|
|
||||||
return (ebig - esmall) / (ebig + esmall);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,15 +0,0 @@
|
|||||||
.global _longjmp
|
|
||||||
.global longjmp
|
|
||||||
|
|
||||||
_longjmp:
|
|
||||||
longjmp:
|
|
||||||
mov 4(%esp),%edx
|
|
||||||
mov 8(%esp),%eax
|
|
||||||
cmp $1,%eax
|
|
||||||
adc $0, %al
|
|
||||||
mov (%edx),%ebx
|
|
||||||
mov 4(%edx),%esi
|
|
||||||
mov 8(%edx),%edi
|
|
||||||
mov 12(%edx),%ebp
|
|
||||||
mov 16(%edx),%esp
|
|
||||||
jmp *20(%edx)
|
|
@ -1,20 +0,0 @@
|
|||||||
.global ___setjmp
|
|
||||||
.global __setjmp
|
|
||||||
.global _setjmp
|
|
||||||
.global setjmp
|
|
||||||
|
|
||||||
___setjmp:
|
|
||||||
__setjmp:
|
|
||||||
_setjmp:
|
|
||||||
setjmp:
|
|
||||||
mov 4(%esp), %eax
|
|
||||||
mov %ebx, (%eax)
|
|
||||||
mov %esi, 4(%eax)
|
|
||||||
mov %edi, 8(%eax)
|
|
||||||
mov %ebp, 12(%eax)
|
|
||||||
lea 4(%esp), %ecx
|
|
||||||
mov %ecx, 16(%eax)
|
|
||||||
mov (%esp), %ecx
|
|
||||||
mov %ecx, 20(%eax)
|
|
||||||
xor %eax, %eax
|
|
||||||
ret
|
|
@ -1,6 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
void clearerr(FILE *stream) {
|
|
||||||
stream->error = 0;
|
|
||||||
stream->eof = 0;
|
|
||||||
}
|
|
@ -1,56 +0,0 @@
|
|||||||
#include <ksys.h>
|
|
||||||
#include "conio.h"
|
|
||||||
|
|
||||||
static char* __con_caption = "Console application";
|
|
||||||
static char* __con_dllname = "/sys/lib/console.obj";
|
|
||||||
|
|
||||||
int __con_is_load = 0;
|
|
||||||
|
|
||||||
unsigned *__con_dll_ver;
|
|
||||||
void stdcall (*__con_init_hidden)(int wnd_width, int wnd_height,int scr_width, int scr_height, const char* title);
|
|
||||||
void stdcall (*__con_write_asciiz)(const char* str);
|
|
||||||
void stdcall (*__con_write_string)(const char* str, unsigned length);
|
|
||||||
int stdcall (*__con_getch)(void);
|
|
||||||
short stdcall (*__con_getch2)(void);
|
|
||||||
int stdcall (*__con_kbhit)(void);
|
|
||||||
char* stdcall (*__con_gets)(char* str, int n);
|
|
||||||
char* stdcall (*__con_gets2)(__con_gets2_callback callback, char* str, int n);
|
|
||||||
void stdcall (*__con_exit)(int status);
|
|
||||||
|
|
||||||
static void __con_lib_link(ksys_coff_etable_t *exp)
|
|
||||||
{
|
|
||||||
__con_dll_ver = _ksys_cofflib_getproc(exp, "con_dll_ver");
|
|
||||||
__con_init_hidden = _ksys_cofflib_getproc(exp, "con_init");
|
|
||||||
__con_write_asciiz = _ksys_cofflib_getproc(exp, "con_write_asciiz");
|
|
||||||
__con_write_string = _ksys_cofflib_getproc(exp, "con_write_string");
|
|
||||||
__con_getch = _ksys_cofflib_getproc(exp, "con_getch");
|
|
||||||
__con_getch2 = _ksys_cofflib_getproc(exp, "con_getch2");
|
|
||||||
__con_kbhit = _ksys_cofflib_getproc(exp, "con_kbhit");
|
|
||||||
__con_gets = _ksys_cofflib_getproc(exp, "con_gets");
|
|
||||||
__con_gets2 = _ksys_cofflib_getproc(exp, "con_gets2");
|
|
||||||
__con_exit = _ksys_cofflib_getproc(exp, "con_exit");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int __con_init(void)
|
|
||||||
{
|
|
||||||
return __con_init_opt(-1, -1, -1, -1, __con_caption);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int __con_init_opt(int wnd_width, int wnd_height,int scr_width, int scr_height, const char* title)
|
|
||||||
{
|
|
||||||
if(!__con_is_load){
|
|
||||||
ksys_coff_etable_t *__con_lib;
|
|
||||||
__con_lib = _ksys_cofflib_load(__con_dllname);
|
|
||||||
if(__con_lib==NULL){
|
|
||||||
_ksys_debug_puts("Error! Can't load console.obj lib\n");
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
__con_lib_link(__con_lib);
|
|
||||||
__con_init_hidden(wnd_width, wnd_height, scr_width, scr_height, title);
|
|
||||||
__con_is_load= 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return 1;
|
|
||||||
}
|
|
@ -1,76 +0,0 @@
|
|||||||
/*
|
|
||||||
|
|
||||||
This is adapded thunk for console.obj sys library
|
|
||||||
Only for internal use in stdio.h
|
|
||||||
|
|
||||||
Adapted for tcc by Siemargl, 2016
|
|
||||||
|
|
||||||
*/
|
|
||||||
#ifndef _CONIO_H_
|
|
||||||
#define _CONIO_H_
|
|
||||||
|
|
||||||
#define cdecl __attribute__ ((cdecl))
|
|
||||||
#define stdcall __attribute__ ((stdcall))
|
|
||||||
|
|
||||||
extern void stdcall (*__con_write_asciiz)(const char* str);
|
|
||||||
/* Display ASCIIZ-string to the console at the current position, shifting
|
|
||||||
the current position. */
|
|
||||||
|
|
||||||
extern void stdcall (*__con_write_string)(const char* str, unsigned length);
|
|
||||||
/* Similar to __con_write_asciiz, but length of the string must be given as a
|
|
||||||
separate parameter */
|
|
||||||
|
|
||||||
extern int stdcall (*__con_getch)(void);
|
|
||||||
/* Get one character from the keyboard.
|
|
||||||
|
|
||||||
For normal characters function returns ASCII-code. For extended
|
|
||||||
characters (eg, Fx, and arrows), first function call returns 0
|
|
||||||
and second call returns the extended code (similar to the DOS-function
|
|
||||||
input). Starting from version 7, after closing the console window,
|
|
||||||
this function returns 0. */
|
|
||||||
|
|
||||||
extern short stdcall (*__con_getch2)(void);
|
|
||||||
/* Reads a character from the keyboard. Low byte contains the ASCII-code
|
|
||||||
(0 for extended characters), high byte - advanced code (like in BIOS
|
|
||||||
input functions). Starting from version 7, after closing the console
|
|
||||||
window, this function returns 0. */
|
|
||||||
|
|
||||||
extern int stdcall (*__con_kbhit)(void);
|
|
||||||
/* Returns 1 if a key was pressed, 0 otherwise. To read pressed keys use
|
|
||||||
__con_getch and __con_getch2. Starting from version 6, after closing
|
|
||||||
the console window, this function returns 1. */
|
|
||||||
|
|
||||||
extern char* stdcall (*__con_gets)(char* str, int n);
|
|
||||||
/* Reads a string from the keyboard. Reading is interrupted when got
|
|
||||||
"new line" character, or after reading the (n-1) characters (depending on
|
|
||||||
what comes first). In the first case the newline is also recorded in the
|
|
||||||
str. The acquired line is complemented by a null character.
|
|
||||||
Starting from version 6, the function returns a pointer to the entered
|
|
||||||
line if reading was successful, and NULL if the console window was closed. */
|
|
||||||
|
|
||||||
typedef int (stdcall * __con_gets2_callback)(int keycode, char** pstr, int* pn,
|
|
||||||
int* ppos);
|
|
||||||
|
|
||||||
extern char* stdcall (*__con_gets2)(__con_gets2_callback callback, char* str, int n);
|
|
||||||
/* Con_gets completely analogous, except that when the user
|
|
||||||
press unrecognized key, it calls the specified callback-procedure
|
|
||||||
(which may, for example, handle up / down for history and tab to enter
|
|
||||||
autocompletion). You should pass to the procedure: key code and three pointers
|
|
||||||
- to the string, to the maximum length and to the current position.
|
|
||||||
function may change the contents of string and may change the string
|
|
||||||
itself (for example, to reallocate memory for increase the limit),
|
|
||||||
maximum length, and position of the line - pointers are passed for it.
|
|
||||||
Return value: 0 = line wasn't changed 1 = line changed, you should
|
|
||||||
remove old string and display new, 2 = line changed, it is necessary
|
|
||||||
to display it; 3 = immediately exit the function.
|
|
||||||
Starting from version 6, the function returns a pointer to the entered
|
|
||||||
line with the successful reading, and NULL if the console window was closed. */
|
|
||||||
|
|
||||||
extern int __con_is_load;
|
|
||||||
extern unsigned *__con_dll_ver;
|
|
||||||
|
|
||||||
extern int __con_init(void);
|
|
||||||
extern int __con_init_opt(int wnd_width, int wnd_height, int scr_width, int scr_height, const char* title);
|
|
||||||
extern void stdcall (*__con_exit)(int status);
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,14 +0,0 @@
|
|||||||
/* Copyright (C) 2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
|
|
||||||
void debug_printf(const char *format,...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
char log_board[300];
|
|
||||||
va_start (ap, format);
|
|
||||||
vsnprintf(log_board, 300, format, ap);
|
|
||||||
va_end(ap);
|
|
||||||
_ksys_debug_puts(log_board);
|
|
||||||
}
|
|
@ -1,7 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
int fclose(FILE *stream) {
|
|
||||||
free(stream);
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int feof(FILE *stream) {
|
|
||||||
return stream->eof;
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int ferror(FILE *stream) {
|
|
||||||
return stream->error;
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int fflush(FILE *stream) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,38 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
int fgetc(FILE* stream)
|
|
||||||
{
|
|
||||||
unsigned bytes_read;
|
|
||||||
char c;
|
|
||||||
|
|
||||||
unsigned status = _ksys_file_read_file(stream->name, stream->position, 1, &c, &bytes_read);
|
|
||||||
|
|
||||||
if (status != KSYS_FS_ERR_SUCCESS) {
|
|
||||||
switch (status) {
|
|
||||||
case KSYS_FS_ERR_EOF:
|
|
||||||
stream->eof = 1;
|
|
||||||
break;
|
|
||||||
case KSYS_FS_ERR_1:
|
|
||||||
case KSYS_FS_ERR_2:
|
|
||||||
case KSYS_FS_ERR_3:
|
|
||||||
case KSYS_FS_ERR_4:
|
|
||||||
case KSYS_FS_ERR_5:
|
|
||||||
case KSYS_FS_ERR_7:
|
|
||||||
case KSYS_FS_ERR_8:
|
|
||||||
case KSYS_FS_ERR_9:
|
|
||||||
case KSYS_FS_ERR_10:
|
|
||||||
case KSYS_FS_ERR_11:
|
|
||||||
default:
|
|
||||||
// Just some IO error, who knows what exactly happened
|
|
||||||
errno = EIO;
|
|
||||||
stream->error = errno;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return EOF;
|
|
||||||
}
|
|
||||||
|
|
||||||
stream->position++;
|
|
||||||
return c;
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int fgetpos(FILE *restrict stream, fpos_t *restrict pos) {
|
|
||||||
*pos = stream->position;
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include "conio.h"
|
|
||||||
#include <errno.h>
|
|
||||||
|
|
||||||
char *fgets(char *str, int n, FILE *stream)
|
|
||||||
{
|
|
||||||
int i=0, sym_code;
|
|
||||||
|
|
||||||
if(!stream || !str){
|
|
||||||
errno = EINVAL;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (i<n-1){
|
|
||||||
sym_code = fgetc(stream);
|
|
||||||
if(sym_code =='\n' || sym_code == EOF){ break; }
|
|
||||||
str[i]=(char)sym_code;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(i<1){ return NULL; }
|
|
||||||
return str;
|
|
||||||
}
|
|
||||||
|
|
@ -1,7 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
FILE *fopen(const char *restrict _name, const char *restrict _mode) {
|
|
||||||
FILE *out = malloc(sizeof(FILE));
|
|
||||||
return freopen(_name, _mode, out);
|
|
||||||
}
|
|
@ -1,816 +0,0 @@
|
|||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// \author (c) Marco Paland (info@paland.com)
|
|
||||||
// 2014-2019, PALANDesign Hannover, Germany
|
|
||||||
//
|
|
||||||
// \license The MIT License (MIT)
|
|
||||||
//
|
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
|
||||||
// in the Software without restriction, including without limitation the rights
|
|
||||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
// copies of the Software, and to permit persons to whom the Software is
|
|
||||||
// furnished to do so, subject to the following conditions:
|
|
||||||
//
|
|
||||||
// The above copyright notice and this permission notice shall be included in
|
|
||||||
// all copies or substantial portions of the Software.
|
|
||||||
//
|
|
||||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
// THE SOFTWARE.
|
|
||||||
//
|
|
||||||
// \brief Tiny printf, sprintf and (v)snprintf implementation, optimized for speed on
|
|
||||||
// embedded systems with a very limited resources. These routines are thread
|
|
||||||
// safe and reentrant!
|
|
||||||
// Use this instead of the bloated standard/newlib printf cause these use
|
|
||||||
// malloc for printf (and may not be thread safe).
|
|
||||||
//
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include "format_print.h"
|
|
||||||
|
|
||||||
// 'ntoa' conversion buffer size, this must be big enough to hold one converted
|
|
||||||
// numeric number including padded zeros (dynamically created on stack)
|
|
||||||
// default: 32 byte
|
|
||||||
#ifndef PRINTF_NTOA_BUFFER_SIZE
|
|
||||||
#define PRINTF_NTOA_BUFFER_SIZE 32U
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// 'ftoa' conversion buffer size, this must be big enough to hold one converted
|
|
||||||
// float number including padded zeros (dynamically created on stack)
|
|
||||||
// default: 32 byte
|
|
||||||
#ifndef PRINTF_FTOA_BUFFER_SIZE
|
|
||||||
#define PRINTF_FTOA_BUFFER_SIZE 32U
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// support for the floating point type (%f)
|
|
||||||
// default: activated
|
|
||||||
#ifndef PRINTF_DISABLE_SUPPORT_FLOAT
|
|
||||||
#define PRINTF_SUPPORT_FLOAT
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// support for exponential floating point notation (%e/%g)
|
|
||||||
// default: activated
|
|
||||||
#ifndef PRINTF_DISABLE_SUPPORT_EXPONENTIAL
|
|
||||||
#define PRINTF_SUPPORT_EXPONENTIAL
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// define the default floating point precision
|
|
||||||
// default: 6 digits
|
|
||||||
#ifndef PRINTF_DEFAULT_FLOAT_PRECISION
|
|
||||||
#define PRINTF_DEFAULT_FLOAT_PRECISION 6U
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// define the largest float suitable to print with %f
|
|
||||||
// default: 1e9
|
|
||||||
#ifndef PRINTF_MAX_FLOAT
|
|
||||||
#define PRINTF_MAX_FLOAT 1e9
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// support for the long long types (%llu or %p)
|
|
||||||
// default: activated
|
|
||||||
#ifndef PRINTF_DISABLE_SUPPORT_LONG_LONG
|
|
||||||
#define PRINTF_SUPPORT_LONG_LONG
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// support for the ptrdiff_t type (%t)
|
|
||||||
// ptrdiff_t is normally defined in <stddef.h> as long or long long type
|
|
||||||
// default: activated
|
|
||||||
#ifndef PRINTF_DISABLE_SUPPORT_PTRDIFF_T
|
|
||||||
#define PRINTF_SUPPORT_PTRDIFF_T
|
|
||||||
#endif
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
// internal flag definitions
|
|
||||||
#define FLAGS_ZEROPAD (1U << 0U)
|
|
||||||
#define FLAGS_LEFT (1U << 1U)
|
|
||||||
#define FLAGS_PLUS (1U << 2U)
|
|
||||||
#define FLAGS_SPACE (1U << 3U)
|
|
||||||
#define FLAGS_HASH (1U << 4U)
|
|
||||||
#define FLAGS_UPPERCASE (1U << 5U)
|
|
||||||
#define FLAGS_CHAR (1U << 6U)
|
|
||||||
#define FLAGS_SHORT (1U << 7U)
|
|
||||||
#define FLAGS_LONG (1U << 8U)
|
|
||||||
#define FLAGS_LONG_LONG (1U << 9U)
|
|
||||||
#define FLAGS_PRECISION (1U << 10U)
|
|
||||||
#define FLAGS_ADAPT_EXP (1U << 11U)
|
|
||||||
|
|
||||||
|
|
||||||
// import float.h for DBL_MAX
|
|
||||||
#if defined(PRINTF_SUPPORT_FLOAT)
|
|
||||||
#include <float.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
// internal buffer output
|
|
||||||
void _out_buffer(char character, void* buffer, size_t idx, size_t maxlen)
|
|
||||||
{
|
|
||||||
if (idx < maxlen) {
|
|
||||||
((char*)buffer)[idx] = character;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal null output
|
|
||||||
void _out_null(char character, void* buffer, size_t idx, size_t maxlen)
|
|
||||||
{
|
|
||||||
(void)character; (void)buffer; (void)idx; (void)maxlen;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal secure strlen
|
|
||||||
// \return The length of the string (excluding the terminating 0) limited by 'maxsize'
|
|
||||||
static inline unsigned int _strnlen_s(const char* str, size_t maxsize)
|
|
||||||
{
|
|
||||||
const char* s;
|
|
||||||
for (s = str; *s && maxsize--; ++s);
|
|
||||||
return (unsigned int)(s - str);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal test if char is a digit (0-9)
|
|
||||||
// \return true if char is a digit
|
|
||||||
static inline bool _is_digit(char ch)
|
|
||||||
{
|
|
||||||
return (ch >= '0') && (ch <= '9');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal ASCII string to unsigned int conversion
|
|
||||||
static unsigned int _atoi(const char** str)
|
|
||||||
{
|
|
||||||
unsigned int i = 0U;
|
|
||||||
while (_is_digit(**str)) {
|
|
||||||
i = i * 10U + (unsigned int)(*((*str)++) - '0');
|
|
||||||
}
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// output the specified string in reverse, taking care of any zero-padding
|
|
||||||
static size_t _out_rev(out_fct_type out, char* buffer, size_t idx, size_t maxlen, const char* buf, size_t len, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
const size_t start_idx = idx;
|
|
||||||
|
|
||||||
// pad spaces up to given width
|
|
||||||
if (!(flags & FLAGS_LEFT) && !(flags & FLAGS_ZEROPAD)) {
|
|
||||||
for (size_t i = len; i < width; i++) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// reverse string
|
|
||||||
while (len) {
|
|
||||||
out(buf[--len], buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
|
|
||||||
// append pad spaces up to given width
|
|
||||||
if (flags & FLAGS_LEFT) {
|
|
||||||
while (idx - start_idx < width) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal itoa format
|
|
||||||
static size_t _ntoa_format(out_fct_type out, char* buffer, size_t idx, size_t maxlen, char* buf, size_t len, bool negative, unsigned int base, unsigned int prec, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
// pad leading zeros
|
|
||||||
if (!(flags & FLAGS_LEFT)) {
|
|
||||||
if (width && (flags & FLAGS_ZEROPAD) && (negative || (flags & (FLAGS_PLUS | FLAGS_SPACE)))) {
|
|
||||||
width--;
|
|
||||||
}
|
|
||||||
while ((len < prec) && (len < PRINTF_NTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
}
|
|
||||||
while ((flags & FLAGS_ZEROPAD) && (len < width) && (len < PRINTF_NTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// handle hash
|
|
||||||
if (flags & FLAGS_HASH) {
|
|
||||||
if (!(flags & FLAGS_PRECISION) && len && ((len == prec) || (len == width))) {
|
|
||||||
len--;
|
|
||||||
if (len && (base == 16U)) {
|
|
||||||
len--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ((base == 16U) && !(flags & FLAGS_UPPERCASE) && (len < PRINTF_NTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = 'x';
|
|
||||||
}
|
|
||||||
else if ((base == 16U) && (flags & FLAGS_UPPERCASE) && (len < PRINTF_NTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = 'X';
|
|
||||||
}
|
|
||||||
else if ((base == 2U) && (len < PRINTF_NTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = 'b';
|
|
||||||
}
|
|
||||||
if (len < PRINTF_NTOA_BUFFER_SIZE) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (len < PRINTF_NTOA_BUFFER_SIZE) {
|
|
||||||
if (negative) {
|
|
||||||
buf[len++] = '-';
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_PLUS) {
|
|
||||||
buf[len++] = '+'; // ignore the space if the '+' exists
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_SPACE) {
|
|
||||||
buf[len++] = ' ';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return _out_rev(out, buffer, idx, maxlen, buf, len, width, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal itoa for 'long' type
|
|
||||||
static size_t _ntoa_long(out_fct_type out, char* buffer, size_t idx, size_t maxlen, unsigned long value, bool negative, unsigned long base, unsigned int prec, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
char buf[PRINTF_NTOA_BUFFER_SIZE];
|
|
||||||
size_t len = 0U;
|
|
||||||
|
|
||||||
// no hash for 0 values
|
|
||||||
if (!value) {
|
|
||||||
flags &= ~FLAGS_HASH;
|
|
||||||
}
|
|
||||||
|
|
||||||
// write if precision != 0 and value is != 0
|
|
||||||
if (!(flags & FLAGS_PRECISION) || value) {
|
|
||||||
do {
|
|
||||||
const char digit = (char)(value % base);
|
|
||||||
buf[len++] = digit < 10 ? '0' + digit : (flags & FLAGS_UPPERCASE ? 'A' : 'a') + digit - 10;
|
|
||||||
value /= base;
|
|
||||||
} while (value && (len < PRINTF_NTOA_BUFFER_SIZE));
|
|
||||||
}
|
|
||||||
|
|
||||||
return _ntoa_format(out, buffer, idx, maxlen, buf, len, negative, (unsigned int)base, prec, width, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// internal itoa for 'long long' type
|
|
||||||
#if defined(PRINTF_SUPPORT_LONG_LONG)
|
|
||||||
static size_t _ntoa_long_long(out_fct_type out, char* buffer, size_t idx, size_t maxlen, unsigned long long value, bool negative, unsigned long long base, unsigned int prec, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
char buf[PRINTF_NTOA_BUFFER_SIZE];
|
|
||||||
size_t len = 0U;
|
|
||||||
|
|
||||||
// no hash for 0 values
|
|
||||||
if (!value) {
|
|
||||||
flags &= ~FLAGS_HASH;
|
|
||||||
}
|
|
||||||
|
|
||||||
// write if precision != 0 and value is != 0
|
|
||||||
if (!(flags & FLAGS_PRECISION) || value) {
|
|
||||||
do {
|
|
||||||
const char digit = (char)(value % base);
|
|
||||||
buf[len++] = digit < 10 ? '0' + digit : (flags & FLAGS_UPPERCASE ? 'A' : 'a') + digit - 10;
|
|
||||||
value /= base;
|
|
||||||
} while (value && (len < PRINTF_NTOA_BUFFER_SIZE));
|
|
||||||
}
|
|
||||||
|
|
||||||
return _ntoa_format(out, buffer, idx, maxlen, buf, len, negative, (unsigned int)base, prec, width, flags);
|
|
||||||
}
|
|
||||||
#endif // PRINTF_SUPPORT_LONG_LONG
|
|
||||||
|
|
||||||
|
|
||||||
#if defined(PRINTF_SUPPORT_FLOAT)
|
|
||||||
|
|
||||||
#if defined(PRINTF_SUPPORT_EXPONENTIAL)
|
|
||||||
// forward declaration so that _ftoa can switch to exp notation for values > PRINTF_MAX_FLOAT
|
|
||||||
static size_t _etoa(out_fct_type out, char* buffer, size_t idx, size_t maxlen, double value, unsigned int prec, unsigned int width, unsigned int flags);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
// internal ftoa for fixed decimal floating point
|
|
||||||
static size_t _ftoa(out_fct_type out, char* buffer, size_t idx, size_t maxlen, double value, unsigned int prec, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
char buf[PRINTF_FTOA_BUFFER_SIZE];
|
|
||||||
size_t len = 0U;
|
|
||||||
double diff = 0.0;
|
|
||||||
|
|
||||||
// powers of 10
|
|
||||||
static const double pow10[] = { 1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000 };
|
|
||||||
|
|
||||||
// test for special values
|
|
||||||
if (value != value)
|
|
||||||
return _out_rev(out, buffer, idx, maxlen, "nan", 3, width, flags);
|
|
||||||
if (value < -DBL_MAX)
|
|
||||||
return _out_rev(out, buffer, idx, maxlen, "fni-", 4, width, flags);
|
|
||||||
if (value > DBL_MAX)
|
|
||||||
return _out_rev(out, buffer, idx, maxlen, (flags & FLAGS_PLUS) ? "fni+" : "fni", (flags & FLAGS_PLUS) ? 4U : 3U, width, flags);
|
|
||||||
|
|
||||||
// test for very large values
|
|
||||||
// standard printf behavior is to print EVERY whole number digit -- which could be 100s of characters overflowing your buffers == bad
|
|
||||||
if ((value > PRINTF_MAX_FLOAT) || (value < -PRINTF_MAX_FLOAT)) {
|
|
||||||
#if defined(PRINTF_SUPPORT_EXPONENTIAL)
|
|
||||||
return _etoa(out, buffer, idx, maxlen, value, prec, width, flags);
|
|
||||||
#else
|
|
||||||
return 0U;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// test for negative
|
|
||||||
bool negative = false;
|
|
||||||
if (value < 0) {
|
|
||||||
negative = true;
|
|
||||||
value = 0 - value;
|
|
||||||
}
|
|
||||||
|
|
||||||
// set default precision, if not set explicitly
|
|
||||||
if (!(flags & FLAGS_PRECISION)) {
|
|
||||||
prec = PRINTF_DEFAULT_FLOAT_PRECISION;
|
|
||||||
}
|
|
||||||
// limit precision to 9, cause a prec >= 10 can lead to overflow errors
|
|
||||||
while ((len < PRINTF_FTOA_BUFFER_SIZE) && (prec > 9U)) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
prec--;
|
|
||||||
}
|
|
||||||
|
|
||||||
int whole = (int)value;
|
|
||||||
double tmp = (value - whole) * pow10[prec];
|
|
||||||
unsigned long frac = (unsigned long)tmp;
|
|
||||||
diff = tmp - frac;
|
|
||||||
|
|
||||||
if (diff > 0.5) {
|
|
||||||
++frac;
|
|
||||||
// handle rollover, e.g. case 0.99 with prec 1 is 1.0
|
|
||||||
if (frac >= pow10[prec]) {
|
|
||||||
frac = 0;
|
|
||||||
++whole;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (diff < 0.5) {
|
|
||||||
}
|
|
||||||
else if ((frac == 0U) || (frac & 1U)) {
|
|
||||||
// if halfway, round up if odd OR if last digit is 0
|
|
||||||
++frac;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (prec == 0U) {
|
|
||||||
diff = value - (double)whole;
|
|
||||||
if ((!(diff < 0.5) || (diff > 0.5)) && (whole & 1)) {
|
|
||||||
// exactly 0.5 and ODD, then round up
|
|
||||||
// 1.5 -> 2, but 2.5 -> 2
|
|
||||||
++whole;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
unsigned int count = prec;
|
|
||||||
// now do fractional part, as an unsigned number
|
|
||||||
while (len < PRINTF_FTOA_BUFFER_SIZE) {
|
|
||||||
--count;
|
|
||||||
buf[len++] = (char)(48U + (frac % 10U));
|
|
||||||
if (!(frac /= 10U)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// add extra 0s
|
|
||||||
while ((len < PRINTF_FTOA_BUFFER_SIZE) && (count-- > 0U)) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
}
|
|
||||||
if (len < PRINTF_FTOA_BUFFER_SIZE) {
|
|
||||||
// add decimal
|
|
||||||
buf[len++] = '.';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// do whole part, number is reversed
|
|
||||||
while (len < PRINTF_FTOA_BUFFER_SIZE) {
|
|
||||||
buf[len++] = (char)(48 + (whole % 10));
|
|
||||||
if (!(whole /= 10)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// pad leading zeros
|
|
||||||
if (!(flags & FLAGS_LEFT) && (flags & FLAGS_ZEROPAD)) {
|
|
||||||
if (width && (negative || (flags & (FLAGS_PLUS | FLAGS_SPACE)))) {
|
|
||||||
width--;
|
|
||||||
}
|
|
||||||
while ((len < width) && (len < PRINTF_FTOA_BUFFER_SIZE)) {
|
|
||||||
buf[len++] = '0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (len < PRINTF_FTOA_BUFFER_SIZE) {
|
|
||||||
if (negative) {
|
|
||||||
buf[len++] = '-';
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_PLUS) {
|
|
||||||
buf[len++] = '+'; // ignore the space if the '+' exists
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_SPACE) {
|
|
||||||
buf[len++] = ' ';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return _out_rev(out, buffer, idx, maxlen, buf, len, width, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#if defined(PRINTF_SUPPORT_EXPONENTIAL)
|
|
||||||
// internal ftoa variant for exponential floating-point type, contributed by Martijn Jasperse <m.jasperse@gmail.com>
|
|
||||||
static size_t _etoa(out_fct_type out, char* buffer, size_t idx, size_t maxlen, double value, unsigned int prec, unsigned int width, unsigned int flags)
|
|
||||||
{
|
|
||||||
// check for NaN and special values
|
|
||||||
if ((value != value) || (value > DBL_MAX) || (value < -DBL_MAX)) {
|
|
||||||
return _ftoa(out, buffer, idx, maxlen, value, prec, width, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
// determine the sign
|
|
||||||
const bool negative = value < 0;
|
|
||||||
if (negative) {
|
|
||||||
value = -value;
|
|
||||||
}
|
|
||||||
|
|
||||||
// default precision
|
|
||||||
if (!(flags & FLAGS_PRECISION)) {
|
|
||||||
prec = PRINTF_DEFAULT_FLOAT_PRECISION;
|
|
||||||
}
|
|
||||||
|
|
||||||
// determine the decimal exponent
|
|
||||||
// based on the algorithm by David Gay (https://www.ampl.com/netlib/fp/dtoa.c)
|
|
||||||
union {
|
|
||||||
uint64_t U;
|
|
||||||
double F;
|
|
||||||
} conv;
|
|
||||||
|
|
||||||
conv.F = value;
|
|
||||||
int exp2 = (int)((conv.U >> 52U) & 0x07FFU) - 1023; // effectively log2
|
|
||||||
conv.U = (conv.U & ((1ULL << 52U) - 1U)) | (1023ULL << 52U); // drop the exponent so conv.F is now in [1,2)
|
|
||||||
// now approximate log10 from the log2 integer part and an expansion of ln around 1.5
|
|
||||||
int expval = (int)(0.1760912590558 + exp2 * 0.301029995663981 + (conv.F - 1.5) * 0.289529654602168);
|
|
||||||
// now we want to compute 10^expval but we want to be sure it won't overflow
|
|
||||||
exp2 = (int)(expval * 3.321928094887362 + 0.5);
|
|
||||||
const double z = expval * 2.302585092994046 - exp2 * 0.6931471805599453;
|
|
||||||
const double z2 = z * z;
|
|
||||||
conv.U = (uint64_t)(exp2 + 1023) << 52U;
|
|
||||||
// compute exp(z) using continued fractions, see https://en.wikipedia.org/wiki/Exponential_function#Continued_fractions_for_ex
|
|
||||||
conv.F *= 1 + 2 * z / (2 - z + (z2 / (6 + (z2 / (10 + z2 / 14)))));
|
|
||||||
// correct for rounding errors
|
|
||||||
if (value < conv.F) {
|
|
||||||
expval--;
|
|
||||||
conv.F /= 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
// the exponent format is "%+03d" and largest value is "307", so set aside 4-5 characters
|
|
||||||
unsigned int minwidth = ((expval < 100) && (expval > -100)) ? 4U : 5U;
|
|
||||||
|
|
||||||
// in "%g" mode, "prec" is the number of *significant figures* not decimals
|
|
||||||
if (flags & FLAGS_ADAPT_EXP) {
|
|
||||||
// do we want to fall-back to "%f" mode?
|
|
||||||
if ((value >= 1e-4) && (value < 1e6)) {
|
|
||||||
if ((int)prec > expval) {
|
|
||||||
prec = (unsigned)((int)prec - expval - 1);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
prec = 0;
|
|
||||||
}
|
|
||||||
flags |= FLAGS_PRECISION; // make sure _ftoa respects precision
|
|
||||||
// no characters in exponent
|
|
||||||
minwidth = 0U;
|
|
||||||
expval = 0;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// we use one sigfig for the whole part
|
|
||||||
if ((prec > 0) && (flags & FLAGS_PRECISION)) {
|
|
||||||
--prec;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// will everything fit?
|
|
||||||
unsigned int fwidth = width;
|
|
||||||
if (width > minwidth) {
|
|
||||||
// we didn't fall-back so subtract the characters required for the exponent
|
|
||||||
fwidth -= minwidth;
|
|
||||||
} else {
|
|
||||||
// not enough characters, so go back to default sizing
|
|
||||||
fwidth = 0U;
|
|
||||||
}
|
|
||||||
if ((flags & FLAGS_LEFT) && minwidth) {
|
|
||||||
// if we're padding on the right, DON'T pad the floating part
|
|
||||||
fwidth = 0U;
|
|
||||||
}
|
|
||||||
|
|
||||||
// rescale the float value
|
|
||||||
if (expval) {
|
|
||||||
value /= conv.F;
|
|
||||||
}
|
|
||||||
|
|
||||||
// output the floating part
|
|
||||||
const size_t start_idx = idx;
|
|
||||||
idx = _ftoa(out, buffer, idx, maxlen, negative ? -value : value, prec, fwidth, flags & ~FLAGS_ADAPT_EXP);
|
|
||||||
|
|
||||||
// output the exponent part
|
|
||||||
if (minwidth) {
|
|
||||||
// output the exponential symbol
|
|
||||||
out((flags & FLAGS_UPPERCASE) ? 'E' : 'e', buffer, idx++, maxlen);
|
|
||||||
// output the exponent value
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, (expval < 0) ? -expval : expval, expval < 0, 10, 0, minwidth-1, FLAGS_ZEROPAD | FLAGS_PLUS);
|
|
||||||
// might need to right-pad spaces
|
|
||||||
if (flags & FLAGS_LEFT) {
|
|
||||||
while (idx - start_idx < width) out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
#endif // PRINTF_SUPPORT_EXPONENTIAL
|
|
||||||
#endif // PRINTF_SUPPORT_FLOAT
|
|
||||||
|
|
||||||
|
|
||||||
// internal vsnprintf
|
|
||||||
int _vsnprintf(out_fct_type out, char* buffer, const size_t maxlen, const char* format, va_list va)
|
|
||||||
{
|
|
||||||
unsigned int flags, width, precision, n;
|
|
||||||
size_t idx = 0U;
|
|
||||||
|
|
||||||
if (!buffer) {
|
|
||||||
// use null output function
|
|
||||||
out = _out_null;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (*format)
|
|
||||||
{
|
|
||||||
// format specifier? %[flags][width][.precision][length]
|
|
||||||
if (*format != '%') {
|
|
||||||
// no
|
|
||||||
out(*format, buffer, idx++, maxlen);
|
|
||||||
format++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// yes, evaluate it
|
|
||||||
format++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// evaluate flags
|
|
||||||
flags = 0U;
|
|
||||||
do {
|
|
||||||
switch (*format) {
|
|
||||||
case '0': flags |= FLAGS_ZEROPAD; format++; n = 1U; break;
|
|
||||||
case '-': flags |= FLAGS_LEFT; format++; n = 1U; break;
|
|
||||||
case '+': flags |= FLAGS_PLUS; format++; n = 1U; break;
|
|
||||||
case ' ': flags |= FLAGS_SPACE; format++; n = 1U; break;
|
|
||||||
case '#': flags |= FLAGS_HASH; format++; n = 1U; break;
|
|
||||||
default : n = 0U; break;
|
|
||||||
}
|
|
||||||
} while (n);
|
|
||||||
|
|
||||||
// evaluate width field
|
|
||||||
width = 0U;
|
|
||||||
if (_is_digit(*format)) {
|
|
||||||
width = _atoi(&format);
|
|
||||||
}
|
|
||||||
else if (*format == '*') {
|
|
||||||
const int w = va_arg(va, int);
|
|
||||||
if (w < 0) {
|
|
||||||
flags |= FLAGS_LEFT; // reverse padding
|
|
||||||
width = (unsigned int)-w;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
width = (unsigned int)w;
|
|
||||||
}
|
|
||||||
format++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// evaluate precision field
|
|
||||||
precision = 0U;
|
|
||||||
if (*format == '.') {
|
|
||||||
flags |= FLAGS_PRECISION;
|
|
||||||
format++;
|
|
||||||
if (_is_digit(*format)) {
|
|
||||||
precision = _atoi(&format);
|
|
||||||
}
|
|
||||||
else if (*format == '*') {
|
|
||||||
const int prec = (int)va_arg(va, int);
|
|
||||||
precision = prec > 0 ? (unsigned int)prec : 0U;
|
|
||||||
format++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// evaluate length field
|
|
||||||
switch (*format) {
|
|
||||||
case 'l' :
|
|
||||||
flags |= FLAGS_LONG;
|
|
||||||
format++;
|
|
||||||
if (*format == 'l') {
|
|
||||||
flags |= FLAGS_LONG_LONG;
|
|
||||||
format++;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'h' :
|
|
||||||
flags |= FLAGS_SHORT;
|
|
||||||
format++;
|
|
||||||
if (*format == 'h') {
|
|
||||||
flags |= FLAGS_CHAR;
|
|
||||||
format++;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#if defined(PRINTF_SUPPORT_PTRDIFF_T)
|
|
||||||
case 't' :
|
|
||||||
flags |= (sizeof(ptrdiff_t) == sizeof(long) ? FLAGS_LONG : FLAGS_LONG_LONG);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
case 'j' :
|
|
||||||
flags |= (sizeof(intmax_t) == sizeof(long) ? FLAGS_LONG : FLAGS_LONG_LONG);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
case 'z' :
|
|
||||||
flags |= (sizeof(size_t) == sizeof(long) ? FLAGS_LONG : FLAGS_LONG_LONG);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// evaluate specifier
|
|
||||||
switch (*format) {
|
|
||||||
case 'd' :
|
|
||||||
case 'i' :
|
|
||||||
case 'u' :
|
|
||||||
case 'x' :
|
|
||||||
case 'X' :
|
|
||||||
case 'o' :
|
|
||||||
case 'b' : {
|
|
||||||
// set the base
|
|
||||||
unsigned int base;
|
|
||||||
if (*format == 'x' || *format == 'X') {
|
|
||||||
base = 16U;
|
|
||||||
}
|
|
||||||
else if (*format == 'o') {
|
|
||||||
base = 8U;
|
|
||||||
}
|
|
||||||
else if (*format == 'b') {
|
|
||||||
base = 2U;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
base = 10U;
|
|
||||||
flags &= ~FLAGS_HASH; // no hash for dec format
|
|
||||||
}
|
|
||||||
// uppercase
|
|
||||||
if (*format == 'X') {
|
|
||||||
flags |= FLAGS_UPPERCASE;
|
|
||||||
}
|
|
||||||
|
|
||||||
// no plus or space flag for u, x, X, o, b
|
|
||||||
if ((*format != 'i') && (*format != 'd')) {
|
|
||||||
flags &= ~(FLAGS_PLUS | FLAGS_SPACE);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ignore '0' flag when precision is given
|
|
||||||
if (flags & FLAGS_PRECISION) {
|
|
||||||
flags &= ~FLAGS_ZEROPAD;
|
|
||||||
}
|
|
||||||
|
|
||||||
// convert the integer
|
|
||||||
if ((*format == 'i') || (*format == 'd')) {
|
|
||||||
// signed
|
|
||||||
if (flags & FLAGS_LONG_LONG) {
|
|
||||||
#if defined(PRINTF_SUPPORT_LONG_LONG)
|
|
||||||
const long long value = va_arg(va, long long);
|
|
||||||
idx = _ntoa_long_long(out, buffer, idx, maxlen, (unsigned long long)(value > 0 ? value : 0 - value), value < 0, base, precision, width, flags);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_LONG) {
|
|
||||||
const long value = va_arg(va, long);
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, (unsigned long)(value > 0 ? value : 0 - value), value < 0, base, precision, width, flags);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
const int value = (flags & FLAGS_CHAR) ? (char)va_arg(va, int) : (flags & FLAGS_SHORT) ? (short int)va_arg(va, int) : va_arg(va, int);
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, (unsigned int)(value > 0 ? value : 0 - value), value < 0, base, precision, width, flags);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// unsigned
|
|
||||||
if (flags & FLAGS_LONG_LONG) {
|
|
||||||
#if defined(PRINTF_SUPPORT_LONG_LONG)
|
|
||||||
idx = _ntoa_long_long(out, buffer, idx, maxlen, va_arg(va, unsigned long long), false, base, precision, width, flags);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
else if (flags & FLAGS_LONG) {
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, va_arg(va, unsigned long), false, base, precision, width, flags);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
const unsigned int value = (flags & FLAGS_CHAR) ? (unsigned char)va_arg(va, unsigned int) : (flags & FLAGS_SHORT) ? (unsigned short int)va_arg(va, unsigned int) : va_arg(va, unsigned int);
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, value, false, base, precision, width, flags);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#if defined(PRINTF_SUPPORT_FLOAT)
|
|
||||||
case 'f' :
|
|
||||||
case 'F' :
|
|
||||||
if (*format == 'F') flags |= FLAGS_UPPERCASE;
|
|
||||||
idx = _ftoa(out, buffer, idx, maxlen, va_arg(va, double), precision, width, flags);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
#if defined(PRINTF_SUPPORT_EXPONENTIAL)
|
|
||||||
case 'e':
|
|
||||||
case 'E':
|
|
||||||
case 'g':
|
|
||||||
case 'G':
|
|
||||||
if ((*format == 'g')||(*format == 'G')) flags |= FLAGS_ADAPT_EXP;
|
|
||||||
if ((*format == 'E')||(*format == 'G')) flags |= FLAGS_UPPERCASE;
|
|
||||||
idx = _etoa(out, buffer, idx, maxlen, va_arg(va, double), precision, width, flags);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
#endif // PRINTF_SUPPORT_EXPONENTIAL
|
|
||||||
#endif // PRINTF_SUPPORT_FLOAT
|
|
||||||
case 'c' : {
|
|
||||||
unsigned int l = 1U;
|
|
||||||
// pre padding
|
|
||||||
if (!(flags & FLAGS_LEFT)) {
|
|
||||||
while (l++ < width) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// char output
|
|
||||||
out((char)va_arg(va, int), buffer, idx++, maxlen);
|
|
||||||
// post padding
|
|
||||||
if (flags & FLAGS_LEFT) {
|
|
||||||
while (l++ < width) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case 's' : {
|
|
||||||
const char* p = va_arg(va, char*);
|
|
||||||
unsigned int l = _strnlen_s(p, precision ? precision : (size_t)-1);
|
|
||||||
// pre padding
|
|
||||||
if (flags & FLAGS_PRECISION) {
|
|
||||||
l = (l < precision ? l : precision);
|
|
||||||
}
|
|
||||||
if (!(flags & FLAGS_LEFT)) {
|
|
||||||
while (l++ < width) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// string output
|
|
||||||
while ((*p != 0) && (!(flags & FLAGS_PRECISION) || precision--)) {
|
|
||||||
out(*(p++), buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
// post padding
|
|
||||||
if (flags & FLAGS_LEFT) {
|
|
||||||
while (l++ < width) {
|
|
||||||
out(' ', buffer, idx++, maxlen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'p' : {
|
|
||||||
width = sizeof(void*) * 2U;
|
|
||||||
flags |= FLAGS_ZEROPAD | FLAGS_UPPERCASE;
|
|
||||||
#if defined(PRINTF_SUPPORT_LONG_LONG)
|
|
||||||
const bool is_ll = sizeof(uintptr_t) == sizeof(long long);
|
|
||||||
if (is_ll) {
|
|
||||||
idx = _ntoa_long_long(out, buffer, idx, maxlen, (uintptr_t)va_arg(va, void*), false, 16U, precision, width, flags);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
#endif
|
|
||||||
idx = _ntoa_long(out, buffer, idx, maxlen, (unsigned long)((uintptr_t)va_arg(va, void*)), false, 16U, precision, width, flags);
|
|
||||||
#if defined(PRINTF_SUPPORT_LONG_LONG)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case '%' :
|
|
||||||
out('%', buffer, idx++, maxlen);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default :
|
|
||||||
out(*format, buffer, idx++, maxlen);
|
|
||||||
format++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// termination
|
|
||||||
out((char)0, buffer, idx < maxlen ? idx : maxlen - 1U, maxlen);
|
|
||||||
|
|
||||||
// return written chars without terminating \0
|
|
||||||
return (int)idx;
|
|
||||||
}
|
|
@ -1,12 +0,0 @@
|
|||||||
#include <stddef.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
void (*fct)(char character, void* arg);
|
|
||||||
void* arg;
|
|
||||||
} out_fct_wrap_type;
|
|
||||||
|
|
||||||
typedef void (*out_fct_type)(char character, void* buffer, size_t idx, size_t maxlen);
|
|
||||||
extern void _out_buffer(char character, void* buffer, size_t idx, size_t maxlen);
|
|
||||||
extern void _out_null(char character, void* buffer, size_t idx, size_t maxlen);
|
|
||||||
extern int _vsnprintf(out_fct_type out, char* buffer, const size_t maxlen, const char* format, va_list va);
|
|
@ -1,403 +0,0 @@
|
|||||||
/*
|
|
||||||
function for format read from any source
|
|
||||||
|
|
||||||
Siemargl formats as http://www.cplusplus.com/reference/cstdio/scanf/, no wchar though
|
|
||||||
http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap05.html is used too
|
|
||||||
|
|
||||||
todo:
|
|
||||||
[characters], [^characters]
|
|
||||||
-%n nothing scanned, filled only if good result
|
|
||||||
-%d, i, u, o, x, p read similar - detect base by prefix 0 or 0x
|
|
||||||
-%a
|
|
||||||
-can overflow unsigned as signed
|
|
||||||
-radix point always '.', no LOCALEs
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <ctype.h>
|
|
||||||
#include <math.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include "format_scan.h"
|
|
||||||
|
|
||||||
static int __try_parse_real(long double *real, int ch, const void *src, void *save, virtual_getc vgetc, virtual_ungetc vungetc)
|
|
||||||
// returns 1 if OK, -1 == EOF, -2 parse broken
|
|
||||||
{
|
|
||||||
int sign = 1, have_digits = 0;
|
|
||||||
long long div;
|
|
||||||
|
|
||||||
if (ch == '+')
|
|
||||||
{
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
} else
|
|
||||||
if (ch == '-')
|
|
||||||
{
|
|
||||||
sign = -1;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
};
|
|
||||||
*real = 0.0;
|
|
||||||
for (;;) // mantissa before point
|
|
||||||
{
|
|
||||||
// test ch is valid
|
|
||||||
if (isdigit(ch))
|
|
||||||
{
|
|
||||||
*real = *real * 10 + ch - '0';
|
|
||||||
have_digits++;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF || isspace(ch)) break; // ok, just finish num
|
|
||||||
} else
|
|
||||||
if (ch == '.' || ch == 'E' || ch == 'e')
|
|
||||||
{
|
|
||||||
break; // ok
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
vungetc(save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (ch != '.' && ch != 'E' && ch != 'e') // ok, just integer part
|
|
||||||
{
|
|
||||||
*real *= sign;
|
|
||||||
if (have_digits)
|
|
||||||
return 1;
|
|
||||||
else
|
|
||||||
return -2;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(ch == '.')
|
|
||||||
{
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
div = 10; // use as divisor
|
|
||||||
for (;;) // mantissa after point
|
|
||||||
{
|
|
||||||
// test ch is valid
|
|
||||||
if (isdigit(ch))
|
|
||||||
{
|
|
||||||
*real += (double)(ch - '0') / div;
|
|
||||||
div *= 10;
|
|
||||||
have_digits++;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF || isspace(ch)) break; // ok, just finish num
|
|
||||||
} else
|
|
||||||
if (ch == 'E' || ch == 'e')
|
|
||||||
{
|
|
||||||
break; // ok
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
vungetc(save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (ch != 'E' && ch != 'e') // ok, real as XX.YY
|
|
||||||
{
|
|
||||||
*real *= sign;
|
|
||||||
if (have_digits)
|
|
||||||
return 1;
|
|
||||||
else
|
|
||||||
return -2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
*real *= sign;
|
|
||||||
// exponent
|
|
||||||
sign = 1;
|
|
||||||
if (ch == '+')
|
|
||||||
{
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
} else
|
|
||||||
if (ch == '-')
|
|
||||||
{
|
|
||||||
sign = -1;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
};
|
|
||||||
div = 0;
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
// test ch is valid
|
|
||||||
if (isdigit(ch))
|
|
||||||
{
|
|
||||||
div = div * 10 + ch - '0';
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF || isspace(ch)) break; // ok, just finish num
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
vungetc(save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
div *= sign;
|
|
||||||
*real *= pow(10, div);
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __try_parse_int(long long *digit, int ch, const void *src, void *save, virtual_getc vgetc, virtual_ungetc vungetc)
|
|
||||||
{
|
|
||||||
int sign = 1, base = 10, have_digits = 0;
|
|
||||||
|
|
||||||
if (ch == '+')
|
|
||||||
{
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
} else
|
|
||||||
if (ch == '-')
|
|
||||||
{
|
|
||||||
sign = -1;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
};
|
|
||||||
|
|
||||||
if (ch == '0') // octal or hex, read next
|
|
||||||
{
|
|
||||||
base = 8;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF || isspace(ch))
|
|
||||||
have_digits++;
|
|
||||||
else
|
|
||||||
if (ch == 'x' || ch == 'X')
|
|
||||||
{
|
|
||||||
base = 16;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF) return EOF;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*digit = 0;
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
// test ch is valid
|
|
||||||
if ((isdigit(ch) && base == 10) ||
|
|
||||||
(isdigit(ch) && base == 8 && ch < '8') ||
|
|
||||||
(isxdigit(ch) && base == 16))
|
|
||||||
{
|
|
||||||
if (base == 16)
|
|
||||||
{
|
|
||||||
if (ch <= '9') ch-= '0';
|
|
||||||
else
|
|
||||||
if (ch <= 'F') ch = 10 + ch - 'A';
|
|
||||||
else
|
|
||||||
ch = 10 + ch - 'a';
|
|
||||||
}
|
|
||||||
else
|
|
||||||
ch -= '0';
|
|
||||||
*digit = *digit * base + ch;
|
|
||||||
have_digits++;
|
|
||||||
ch = vgetc(save, src);
|
|
||||||
if (ch == EOF || isspace(ch)) break; // ok, just finish num
|
|
||||||
}
|
|
||||||
else if (ch == EOF || isspace(ch))
|
|
||||||
break;
|
|
||||||
else
|
|
||||||
{
|
|
||||||
vungetc(save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*digit *= sign;
|
|
||||||
if (have_digits)
|
|
||||||
return 1;
|
|
||||||
else
|
|
||||||
return -2;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int _format_scan(const void *src, const char *fmt, va_list argp, virtual_getc vgetc, virtual_ungetc vungetc)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
int length;
|
|
||||||
int fmt1, fmt2; // width, precision
|
|
||||||
size_t pos, posc;
|
|
||||||
const char *fmtc; // first point to %, fmtc points to specifier
|
|
||||||
int ch;
|
|
||||||
int format_flag;
|
|
||||||
int flag_long; // 2 = long double or long long int or wchar
|
|
||||||
int *point_to_n = NULL, nread = 0;
|
|
||||||
int flags; // parsed flags
|
|
||||||
int save = 0;
|
|
||||||
char *arg_str;
|
|
||||||
int *arg_int;
|
|
||||||
long *arg_long;
|
|
||||||
long long *arg_longlong;
|
|
||||||
float *arg_float;
|
|
||||||
double *arg_double;
|
|
||||||
long double *arg_longdouble;
|
|
||||||
long long digit;
|
|
||||||
long double real;
|
|
||||||
|
|
||||||
|
|
||||||
pos = 0;
|
|
||||||
while(*fmt)
|
|
||||||
{
|
|
||||||
while (*fmt && isspace(*fmt)) fmt++; // skip paces in format str
|
|
||||||
|
|
||||||
if (*fmt != '%') // usual char
|
|
||||||
{
|
|
||||||
ch = vgetc(&save, src);
|
|
||||||
if (ch != *fmt++) // char not match format
|
|
||||||
{
|
|
||||||
vungetc(&save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
pos++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (*(fmt + 1) == '%') // %%
|
|
||||||
{
|
|
||||||
ch = vgetc(&save, src);
|
|
||||||
if (ch != '%') // char not match format
|
|
||||||
{
|
|
||||||
vungetc(&save, ch, src);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
pos++;
|
|
||||||
fmt += 2;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
//checking to containg format in the string
|
|
||||||
fmtc = fmt;
|
|
||||||
posc = pos;
|
|
||||||
|
|
||||||
flags = 0;
|
|
||||||
format_flag = 0;
|
|
||||||
flag_long = 0; // 2 = long double or long long int or wchar
|
|
||||||
|
|
||||||
while(*fmtc != '\0' && !format_flag) // searching end of format
|
|
||||||
{
|
|
||||||
fmtc++; posc++;
|
|
||||||
switch( *fmtc )
|
|
||||||
{
|
|
||||||
case 'a':
|
|
||||||
format_flag = 1;
|
|
||||||
flags |= flag_unsigned;
|
|
||||||
break;
|
|
||||||
case 'c': case 'd': case 'i': case 'e': case 'f': case 'g': case 's': case 'n':
|
|
||||||
format_flag = 1;
|
|
||||||
break;
|
|
||||||
case 'l':
|
|
||||||
flag_long = flag_long ? 2 : 1; // ll.eq.L
|
|
||||||
break;
|
|
||||||
case 'L':
|
|
||||||
flag_long = 2;
|
|
||||||
break;
|
|
||||||
case 'o': case 'u': case 'x': case 'p':
|
|
||||||
format_flag = 1;
|
|
||||||
flags |= flag_unsigned;
|
|
||||||
break;
|
|
||||||
case '*': case '.': // just skip
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
if(isdigit(*fmtc)) break;
|
|
||||||
goto exit_me; // non format char found - user error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (format_flag == 0)
|
|
||||||
{
|
|
||||||
goto exit_me; // format char not found - user error
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt1 = 0;
|
|
||||||
fmt2 = 0;
|
|
||||||
if (posc - pos > 1) // try to read width, precision
|
|
||||||
{
|
|
||||||
fmt++;
|
|
||||||
for(i = pos + 1; i < posc; i++)
|
|
||||||
{
|
|
||||||
switch(*fmt)
|
|
||||||
{
|
|
||||||
case '0':
|
|
||||||
if(fmt1 == 0 && (flags & flag_point) == 0) flags |= flag_lead_zeros;
|
|
||||||
case '1': case '2': case '3': case '4':
|
|
||||||
case '5': case '6': case '7': case '8': case '9':
|
|
||||||
if ((flags & flag_point) == 0)
|
|
||||||
fmt1 = fmt1 * 10 + (*fmt -'0');
|
|
||||||
else
|
|
||||||
fmt2 = fmt2 * 10 + (*fmt -'0');
|
|
||||||
break;
|
|
||||||
case '*': // ignoring
|
|
||||||
break;
|
|
||||||
case '.':
|
|
||||||
flags |= flag_point;
|
|
||||||
break;
|
|
||||||
case 'l': case 'L': // valid chars - skip
|
|
||||||
break;
|
|
||||||
default: // must be error
|
|
||||||
goto exit_me; // format char not found - user error
|
|
||||||
}
|
|
||||||
fmt++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// do real work - format arguments values
|
|
||||||
// skip input spaces
|
|
||||||
do {
|
|
||||||
ch = vgetc(&save, src);
|
|
||||||
if (ch == EOF) goto exit_me;
|
|
||||||
} while (isspace(ch));
|
|
||||||
|
|
||||||
switch(*fmtc)
|
|
||||||
{
|
|
||||||
case 'n':
|
|
||||||
point_to_n = va_arg(argp, int*);
|
|
||||||
vungetc(&save, ch, src);
|
|
||||||
break;
|
|
||||||
case 'c': // read width chars, ever spaces
|
|
||||||
arg_str = va_arg(argp, char*);
|
|
||||||
if (fmt1 == 0) length = 1;
|
|
||||||
else length = fmt1;
|
|
||||||
for (i = 0; i < length;)
|
|
||||||
{
|
|
||||||
*arg_str++ = ch; i++;
|
|
||||||
ch = vgetc(&save, src);
|
|
||||||
if (ch == EOF) break;
|
|
||||||
}
|
|
||||||
if (i < length) goto exit_me; // not enough chars
|
|
||||||
break;
|
|
||||||
case 's':
|
|
||||||
arg_str = va_arg(argp, char*);
|
|
||||||
if (fmt1 == 0) length = 4095; // max string scan 4096
|
|
||||||
else length = fmt1;
|
|
||||||
for (i = 0; i < length; i++)
|
|
||||||
{
|
|
||||||
*arg_str++ = ch;
|
|
||||||
ch = vgetc(&save, src);
|
|
||||||
if (ch == EOF || isspace(ch)) break; // ok, just finish string
|
|
||||||
}
|
|
||||||
*arg_str++ = '\0';
|
|
||||||
break;
|
|
||||||
case 'd': case 'i': case 'u':
|
|
||||||
case 'o': case 'p': case 'x':
|
|
||||||
i = __try_parse_int(&digit, ch, src, &save, vgetc, vungetc);
|
|
||||||
if (i < 0) goto exit_me;
|
|
||||||
|
|
||||||
if (flag_long == 0) { arg_int = va_arg(argp, int*); *arg_int = (int)digit; } else
|
|
||||||
if (flag_long == 1) { arg_long = va_arg(argp, long*); *arg_long = (long)digit; } else
|
|
||||||
if (flag_long == 2) { arg_longlong = va_arg(argp, long long*); *arg_longlong = digit; }
|
|
||||||
break;
|
|
||||||
case 'a': case 'A': case 'f': case 'F':
|
|
||||||
case 'e': case 'E':
|
|
||||||
case 'g': case 'G':
|
|
||||||
i = __try_parse_real(&real, ch, src, &save, vgetc, vungetc);
|
|
||||||
if (i < 0) goto exit_me;
|
|
||||||
|
|
||||||
if (flag_long == 0) { arg_float = va_arg(argp, float*); *arg_float = (float)real; } else
|
|
||||||
if (flag_long == 1) { arg_double = va_arg(argp, double*); *arg_double = (double)real; } else
|
|
||||||
if (flag_long == 2) { arg_longdouble = va_arg(argp, long double*); *arg_longdouble = real; }
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt = fmtc + 1;
|
|
||||||
nread++;
|
|
||||||
}
|
|
||||||
exit_me:
|
|
||||||
if (point_to_n) *point_to_n = nread;
|
|
||||||
return nread;
|
|
||||||
}
|
|
@ -1,20 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
typedef int (*virtual_getc)(void *sp, const void *obj);
|
|
||||||
typedef void (*virtual_ungetc)(void *sp, int c, const void *obj);
|
|
||||||
|
|
||||||
enum flags_t
|
|
||||||
{
|
|
||||||
flag_unsigned = 0x02,
|
|
||||||
flag_register = 0x04,
|
|
||||||
flag_plus = 0x08,
|
|
||||||
flag_left_just = 0x10,
|
|
||||||
flag_lead_zeros = 0x20,
|
|
||||||
flag_space_plus = 0x40,
|
|
||||||
flag_hash_sign = 0x80,
|
|
||||||
flag_point = 0x100
|
|
||||||
};
|
|
||||||
|
|
||||||
extern char *__scanf_buffer;
|
|
||||||
|
|
||||||
int _format_scan(const void *src, const char *fmt, va_list argp, virtual_getc vgetc, virtual_ungetc vungetc);
|
|
@ -1,35 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
int fputc(int c, FILE *stream)
|
|
||||||
{
|
|
||||||
unsigned bytes_written;
|
|
||||||
|
|
||||||
unsigned status = _ksys_file_write_file(stream->name, stream->position, 1, &c, &bytes_written);
|
|
||||||
|
|
||||||
if (status != KSYS_FS_ERR_SUCCESS) {
|
|
||||||
switch (status) {
|
|
||||||
case KSYS_FS_ERR_1:
|
|
||||||
case KSYS_FS_ERR_2:
|
|
||||||
case KSYS_FS_ERR_3:
|
|
||||||
case KSYS_FS_ERR_4:
|
|
||||||
case KSYS_FS_ERR_5:
|
|
||||||
case KSYS_FS_ERR_EOF:
|
|
||||||
case KSYS_FS_ERR_7:
|
|
||||||
case KSYS_FS_ERR_8:
|
|
||||||
case KSYS_FS_ERR_9:
|
|
||||||
case KSYS_FS_ERR_10:
|
|
||||||
case KSYS_FS_ERR_11:
|
|
||||||
default:
|
|
||||||
// Just some IO error, who knows what exactly happened
|
|
||||||
errno = EIO;
|
|
||||||
stream->error = errno;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return EOF;
|
|
||||||
}
|
|
||||||
|
|
||||||
stream->position++;
|
|
||||||
return c;
|
|
||||||
}
|
|
@ -1,11 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
int fputs(const char *str, FILE *stream){
|
|
||||||
for(size_t i = 0; i < strlen(str); i++){
|
|
||||||
if (fputc(str[i], stream) == EOF) {
|
|
||||||
return EOF;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,20 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
size_t fread(void *restrict ptr, size_t size, size_t nmemb, FILE *restrict stream) {
|
|
||||||
unsigned bytes_read = 0;
|
|
||||||
unsigned bytes_count = size * nmemb;
|
|
||||||
|
|
||||||
for (size_t i = 0; i < bytes_count; i++) {
|
|
||||||
char c = fgetc(stream);
|
|
||||||
|
|
||||||
if (c == EOF) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
*(char*)(ptr+i) = c;
|
|
||||||
|
|
||||||
bytes_read++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return bytes_read / size;
|
|
||||||
}
|
|
@ -1,36 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
FILE *freopen(const char *restrict _name, const char *restrict _mode, FILE *restrict out) {
|
|
||||||
ksys_bdfe_t info;
|
|
||||||
if (_ksys_file_get_info(_name, &info)) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!out) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
out->name = strdup(_name);
|
|
||||||
out->position = 0;
|
|
||||||
out->error = 0;
|
|
||||||
out->eof = 0;
|
|
||||||
out->kind = 0;
|
|
||||||
out->orientation = 0;
|
|
||||||
out->mode = 0;
|
|
||||||
|
|
||||||
if (strchr(_mode, 'b')) { out->mode |= _STDIO_F_B; }
|
|
||||||
if (strchr(_mode, 'x')) { out->mode |= _STDIO_F_X; }
|
|
||||||
if (strchr(_mode, 'a')) { out->mode |= _STDIO_F_A; }
|
|
||||||
if (strchr(_mode, 'r')) { out->mode |= _STDIO_F_R; }
|
|
||||||
if (strchr(_mode, 'w')) { out->mode |= _STDIO_F_W; }
|
|
||||||
if (strchr(_mode, '+')) { out->mode |= _STDIO_F_R | _STDIO_F_W; }
|
|
||||||
|
|
||||||
if (out->mode & _STDIO_F_A) {
|
|
||||||
out->position = info.size;
|
|
||||||
out->append_offset = info.size;
|
|
||||||
}
|
|
||||||
|
|
||||||
return out;
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int fscanf(FILE* stream, const char* format, ...)
|
|
||||||
{
|
|
||||||
va_list arg;
|
|
||||||
int n;
|
|
||||||
va_start(arg, format);
|
|
||||||
n = vfscanf(stream, format, arg);
|
|
||||||
va_end(arg);
|
|
||||||
return n;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
@ -1,17 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
int fseek(FILE *stream, long int offset, int whence) {
|
|
||||||
if (whence == SEEK_SET) {
|
|
||||||
stream->position = offset;
|
|
||||||
} else if (whence == SEEK_CUR) {
|
|
||||||
stream->position += offset;
|
|
||||||
} else if (whence == SEEK_END) {
|
|
||||||
ksys_bdfe_t info;
|
|
||||||
if (_ksys_file_get_info(stream->name, &info)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
stream->position = info.size + offset;
|
|
||||||
}
|
|
||||||
stream->eof = 0;
|
|
||||||
}
|
|
@ -1,7 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
int fsetpos(FILE *stream, const fpos_t *pos) {
|
|
||||||
stream->position = *pos;
|
|
||||||
stream->eof = 0;
|
|
||||||
return 0;
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
long int ftell(FILE *stream) {
|
|
||||||
return stream->position;
|
|
||||||
}
|
|
@ -1,17 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
size_t fwrite(const void *restrict ptr, size_t size, size_t nmemb, FILE *restrict stream) {
|
|
||||||
unsigned bytes_written = 0;
|
|
||||||
unsigned bytes_count = size * nmemb;
|
|
||||||
|
|
||||||
for (size_t i = 0; i < bytes_count; i++) {
|
|
||||||
char c = *(char*)(ptr+i);
|
|
||||||
if (fputc(c, stream) != c) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
bytes_written++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return bytes_written / size;
|
|
||||||
}
|
|
@ -1,10 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include "conio.h"
|
|
||||||
|
|
||||||
int getchar(void) {
|
|
||||||
int c = __con_getch();
|
|
||||||
if (c == 0) {
|
|
||||||
c = EOF;
|
|
||||||
}
|
|
||||||
return c;
|
|
||||||
}
|
|
@ -1,15 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include "conio.h"
|
|
||||||
#include <errno.h>
|
|
||||||
|
|
||||||
char *gets(char* str)
|
|
||||||
{
|
|
||||||
__con_init();
|
|
||||||
if(__con_gets(str, 4096)==NULL){
|
|
||||||
errno = EIO;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
str[strlen(str)-1]='\0';
|
|
||||||
return str;
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
void perror(const char *s) {
|
|
||||||
printf("%s: It's some error, maybe...", s);
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
/* Copyright (C) 2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
#include "format_print.h"
|
|
||||||
|
|
||||||
int printf(const char *format, ...)
|
|
||||||
{
|
|
||||||
va_list arg;
|
|
||||||
va_start(arg, format);
|
|
||||||
return vprintf(format, arg);
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
/* Copyright (C) 2021 Logaev Maxim (turbocat2001), GPLv2 */
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include "conio.h"
|
|
||||||
|
|
||||||
int puts(const char *str)
|
|
||||||
{
|
|
||||||
__con_init();
|
|
||||||
__con_write_asciiz(str);
|
|
||||||
__con_write_asciiz("\n");
|
|
||||||
return strlen(str);
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
int remove(const char *name) {
|
|
||||||
return _ksys_file_delete(name);
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
#include <ksys.h>
|
|
||||||
|
|
||||||
int rename(const char *name, const char *new_name) {
|
|
||||||
return _ksys_file_rename(name, new_name);
|
|
||||||
}
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user