libc/picolibc: Split hooks into separate files
This splits the picolibc helper functions into separate files instead of smashing them all together. Signed-off-by: Keith Packard <keithp@keithp.com>
This commit is contained in:
parent
e4b830fe7a
commit
c759b8ac3d
|
@ -144,10 +144,10 @@ MEMORY
|
|||
#endif /* CONFIG_NEWLIB_LIBC */
|
||||
|
||||
#ifdef CONFIG_PICOLIBC
|
||||
/* For Picolibc libc-hook.c. */
|
||||
/* For Picolibc, all files under lib/libc/picolibc */
|
||||
#define LIB_C_IN_SECT(lsect) \
|
||||
*liblib__libc__picolibc.a:libc-hooks.c.obj(.##lsect) \
|
||||
*liblib__libc__picolibc.a:libc-hooks.c.obj(.##lsect##.*)
|
||||
*liblib__libc__picolibc.a:(.##lsect) \
|
||||
*liblib__libc__picolibc.a:(.##lsect##.*)
|
||||
|
||||
#endif /* CONFIG_PICOLIBC */
|
||||
|
||||
|
|
|
@ -1,10 +1,15 @@
|
|||
# SPDX-License-Identifier: Apache-2.0
|
||||
|
||||
zephyr_library()
|
||||
zephyr_library_sources(libc-hooks.c)
|
||||
|
||||
# Do not allow LTO when compiling libc-hooks.c file
|
||||
set_source_files_properties(libc-hooks.c PROPERTIES COMPILE_OPTIONS $<TARGET_PROPERTY:compiler,prohibit_lto>)
|
||||
zephyr_library_sources(
|
||||
assert.c
|
||||
cbprintf.c
|
||||
chk_fail.c
|
||||
errno_wrap.c
|
||||
exit.c
|
||||
locks.c
|
||||
stdio.c
|
||||
)
|
||||
|
||||
# define __LINUX_ERRNO_EXTENSIONS__ so we get errno defines like -ESHUTDOWN
|
||||
# used by the network stack
|
||||
|
|
|
@ -0,0 +1,28 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
#ifdef CONFIG_PICOLIBC_ASSERT_VERBOSE
|
||||
|
||||
FUNC_NORETURN void __assert_func(const char *file, int line,
|
||||
const char *function, const char *expression)
|
||||
{
|
||||
__ASSERT(0, "assertion \"%s\" failed: file \"%s\", line %d%s%s\n",
|
||||
expression, file, line,
|
||||
function ? ", function: " : "", function ? function : "");
|
||||
CODE_UNREACHABLE;
|
||||
}
|
||||
|
||||
#else
|
||||
|
||||
FUNC_NORETURN void __assert_no_args(void)
|
||||
{
|
||||
__ASSERT_NO_MSG(0);
|
||||
CODE_UNREACHABLE;
|
||||
}
|
||||
|
||||
#endif
|
|
@ -0,0 +1,31 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
struct cb_bits {
|
||||
FILE f;
|
||||
cbprintf_cb out;
|
||||
void *ctx;
|
||||
};
|
||||
|
||||
static int cbputc(char c, FILE *_s)
|
||||
{
|
||||
struct cb_bits *s = (struct cb_bits *) _s;
|
||||
|
||||
(*s->out) (c, s->ctx);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int cbvprintf(cbprintf_cb out, void *ctx, const char *fp, va_list ap)
|
||||
{
|
||||
struct cb_bits s = {
|
||||
.f = FDEV_SETUP_STREAM(cbputc, NULL, NULL, _FDEV_SETUP_WRITE),
|
||||
.out = out,
|
||||
.ctx = ctx,
|
||||
};
|
||||
return vfprintf(&s.f, fp, ap);
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
/* This function gets called if static buffer overflow detection is enabled on
|
||||
* stdlib side (Picolibc here), in case such an overflow is detected. Picolibc
|
||||
* provides an implementation not suitable for us, so we override it here.
|
||||
*/
|
||||
__weak FUNC_NORETURN void __chk_fail(void)
|
||||
{
|
||||
printk("* buffer overflow detected *\n");
|
||||
z_except_reason(K_ERR_STACK_CHK_FAIL);
|
||||
CODE_UNREACHABLE;
|
||||
}
|
|
@ -0,0 +1,21 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
#ifndef CONFIG_LIBC_ERRNO
|
||||
|
||||
/*
|
||||
* Picolibc needs to be able to declare this itself so that the library
|
||||
* doesn't end up needing zephyr header files. That means using a regular
|
||||
* function instead of an inline.
|
||||
*/
|
||||
int *z_errno_wrap(void)
|
||||
{
|
||||
return z_errno();
|
||||
}
|
||||
|
||||
#endif
|
|
@ -0,0 +1,15 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
__weak void _exit(int status)
|
||||
{
|
||||
printf("exit\n");
|
||||
while (1) {
|
||||
Z_SPIN_DELAY(100);
|
||||
}
|
||||
}
|
|
@ -1,253 +0,0 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <zephyr/posix/sys/stat.h>
|
||||
#include <sys/time.h>
|
||||
#include <zephyr/arch/cpu.h>
|
||||
#include <zephyr/linker/linker-defs.h>
|
||||
#include <zephyr/sys/util.h>
|
||||
#include <zephyr/sys/errno_private.h>
|
||||
#include <zephyr/sys/libc-hooks.h>
|
||||
#include <zephyr/internal/syscall_handler.h>
|
||||
#include <zephyr/app_memory/app_memdomain.h>
|
||||
#include <zephyr/init.h>
|
||||
#include <zephyr/sys/sem.h>
|
||||
#include <zephyr/logging/log.h>
|
||||
#ifdef CONFIG_MMU
|
||||
#include <zephyr/kernel/mm.h>
|
||||
#endif
|
||||
|
||||
#define LIBC_BSS K_APP_BMEM(z_libc_partition)
|
||||
#define LIBC_DATA K_APP_DMEM(z_libc_partition)
|
||||
|
||||
static LIBC_DATA int (*_stdout_hook)(int);
|
||||
|
||||
int z_impl_zephyr_fputc(int a, FILE *out)
|
||||
{
|
||||
(*_stdout_hook)(a);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_USERSPACE
|
||||
static inline int z_vrfy_zephyr_fputc(int c, FILE *stream)
|
||||
{
|
||||
return z_impl_zephyr_fputc(c, stream);
|
||||
}
|
||||
#include <zephyr/syscalls/zephyr_fputc_mrsh.c>
|
||||
#endif
|
||||
|
||||
static int picolibc_put(char a, FILE *f)
|
||||
{
|
||||
zephyr_fputc(a, f);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static LIBC_DATA FILE __stdout = FDEV_SETUP_STREAM(picolibc_put, NULL, NULL, 0);
|
||||
static LIBC_DATA FILE __stdin = FDEV_SETUP_STREAM(NULL, NULL, NULL, 0);
|
||||
|
||||
#ifdef __strong_reference
|
||||
#define STDIO_ALIAS(x) __strong_reference(stdout, x);
|
||||
#else
|
||||
#define STDIO_ALIAS(x) FILE *const x = &__stdout;
|
||||
#endif
|
||||
|
||||
FILE *const stdin = &__stdin;
|
||||
FILE *const stdout = &__stdout;
|
||||
STDIO_ALIAS(stderr);
|
||||
|
||||
void __stdout_hook_install(int (*hook)(int))
|
||||
{
|
||||
_stdout_hook = hook;
|
||||
__stdout.flags |= _FDEV_SETUP_WRITE;
|
||||
}
|
||||
|
||||
void __stdin_hook_install(unsigned char (*hook)(void))
|
||||
{
|
||||
__stdin.get = (int (*)(FILE *)) hook;
|
||||
__stdin.flags |= _FDEV_SETUP_READ;
|
||||
}
|
||||
|
||||
#include <zephyr/sys/cbprintf.h>
|
||||
|
||||
struct cb_bits {
|
||||
FILE f;
|
||||
cbprintf_cb out;
|
||||
void *ctx;
|
||||
};
|
||||
|
||||
static int cbputc(char c, FILE *_s)
|
||||
{
|
||||
struct cb_bits *s = (struct cb_bits *) _s;
|
||||
|
||||
(*s->out) (c, s->ctx);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int cbvprintf(cbprintf_cb out, void *ctx, const char *fp, va_list ap)
|
||||
{
|
||||
struct cb_bits s = {
|
||||
.f = FDEV_SETUP_STREAM(cbputc, NULL, NULL, _FDEV_SETUP_WRITE),
|
||||
.out = out,
|
||||
.ctx = ctx,
|
||||
};
|
||||
return vfprintf(&s.f, fp, ap);
|
||||
}
|
||||
|
||||
__weak void _exit(int status)
|
||||
{
|
||||
printk("exit\n");
|
||||
while (1) {
|
||||
Z_SPIN_DELAY(100);
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef CONFIG_MULTITHREADING
|
||||
#define _LOCK_T void *
|
||||
K_MUTEX_DEFINE(__lock___libc_recursive_mutex);
|
||||
|
||||
#ifdef CONFIG_USERSPACE
|
||||
/* Grant public access to picolibc lock after boot */
|
||||
static int picolibc_locks_prepare(void)
|
||||
{
|
||||
|
||||
/* Initialise recursive locks */
|
||||
k_object_access_all_grant(&__lock___libc_recursive_mutex);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
SYS_INIT(picolibc_locks_prepare, POST_KERNEL,
|
||||
CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|
||||
#endif /* CONFIG_USERSPACE */
|
||||
|
||||
/* Create a new dynamic recursive lock */
|
||||
void __retarget_lock_init_recursive(_LOCK_T *lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
|
||||
/* Allocate mutex object */
|
||||
#ifndef CONFIG_USERSPACE
|
||||
*lock = malloc(sizeof(struct k_mutex));
|
||||
#else
|
||||
*lock = k_object_alloc(K_OBJ_MUTEX);
|
||||
#endif /* !CONFIG_USERSPACE */
|
||||
__ASSERT(*lock != NULL, "recursive lock allocation failed");
|
||||
|
||||
k_mutex_init((struct k_mutex *)*lock);
|
||||
}
|
||||
|
||||
/* Create a new dynamic non-recursive lock */
|
||||
void __retarget_lock_init(_LOCK_T *lock)
|
||||
{
|
||||
__retarget_lock_init_recursive(lock);
|
||||
}
|
||||
|
||||
/* Close dynamic recursive lock */
|
||||
void __retarget_lock_close_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
#ifndef CONFIG_USERSPACE
|
||||
free(lock);
|
||||
#else
|
||||
k_object_release(lock);
|
||||
#endif /* !CONFIG_USERSPACE */
|
||||
}
|
||||
|
||||
/* Close dynamic non-recursive lock */
|
||||
void __retarget_lock_close(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_close_recursive(lock);
|
||||
}
|
||||
|
||||
/* Acquiure recursive lock */
|
||||
void __retarget_lock_acquire_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
k_mutex_lock((struct k_mutex *)lock, K_FOREVER);
|
||||
}
|
||||
|
||||
/* Acquiure non-recursive lock */
|
||||
void __retarget_lock_acquire(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_acquire_recursive(lock);
|
||||
}
|
||||
|
||||
/* Try acquiring recursive lock */
|
||||
int __retarget_lock_try_acquire_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
return !k_mutex_lock((struct k_mutex *)lock, K_NO_WAIT);
|
||||
}
|
||||
|
||||
/* Try acquiring non-recursive lock */
|
||||
int __retarget_lock_try_acquire(_LOCK_T lock)
|
||||
{
|
||||
return __retarget_lock_try_acquire_recursive(lock);
|
||||
}
|
||||
|
||||
/* Release recursive lock */
|
||||
void __retarget_lock_release_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
k_mutex_unlock((struct k_mutex *)lock);
|
||||
}
|
||||
|
||||
/* Release non-recursive lock */
|
||||
void __retarget_lock_release(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_release_recursive(lock);
|
||||
}
|
||||
|
||||
#endif /* CONFIG_MULTITHREADING */
|
||||
|
||||
#ifdef CONFIG_PICOLIBC_ASSERT_VERBOSE
|
||||
|
||||
FUNC_NORETURN void __assert_func(const char *file, int line,
|
||||
const char *function, const char *expression)
|
||||
{
|
||||
__ASSERT(0, "assertion \"%s\" failed: file \"%s\", line %d%s%s\n",
|
||||
expression, file, line,
|
||||
function ? ", function: " : "", function ? function : "");
|
||||
CODE_UNREACHABLE;
|
||||
}
|
||||
|
||||
#else
|
||||
|
||||
FUNC_NORETURN void __assert_no_args(void)
|
||||
{
|
||||
__ASSERT_NO_MSG(0);
|
||||
CODE_UNREACHABLE;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/* This function gets called if static buffer overflow detection is enabled on
|
||||
* stdlib side (Picolibc here), in case such an overflow is detected. Picolibc
|
||||
* provides an implementation not suitable for us, so we override it here.
|
||||
*/
|
||||
__weak FUNC_NORETURN void __chk_fail(void)
|
||||
{
|
||||
printk("* buffer overflow detected *\n");
|
||||
z_except_reason(K_ERR_STACK_CHK_FAIL);
|
||||
CODE_UNREACHABLE;
|
||||
}
|
||||
|
||||
#ifndef CONFIG_LIBC_ERRNO
|
||||
|
||||
/*
|
||||
* Picolibc needs to be able to declare this itself so that the library
|
||||
* doesn't end up needing zephyr header files. That means using a regular
|
||||
* function instead of an inline.
|
||||
*/
|
||||
int *z_errno_wrap(void)
|
||||
{
|
||||
return z_errno();
|
||||
}
|
||||
|
||||
#endif
|
|
@ -0,0 +1,106 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
#ifdef CONFIG_MULTITHREADING
|
||||
#define _LOCK_T void *
|
||||
K_MUTEX_DEFINE(__lock___libc_recursive_mutex);
|
||||
|
||||
#ifdef CONFIG_USERSPACE
|
||||
/* Grant public access to picolibc lock after boot */
|
||||
static int picolibc_locks_prepare(void)
|
||||
{
|
||||
|
||||
/* Initialise recursive locks */
|
||||
k_object_access_all_grant(&__lock___libc_recursive_mutex);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
SYS_INIT(picolibc_locks_prepare, POST_KERNEL,
|
||||
CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|
||||
#endif /* CONFIG_USERSPACE */
|
||||
|
||||
/* Create a new dynamic recursive lock */
|
||||
void __retarget_lock_init_recursive(_LOCK_T *lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
|
||||
/* Allocate mutex object */
|
||||
#ifndef CONFIG_USERSPACE
|
||||
*lock = malloc(sizeof(struct k_mutex));
|
||||
#else
|
||||
*lock = k_object_alloc(K_OBJ_MUTEX);
|
||||
#endif /* !CONFIG_USERSPACE */
|
||||
__ASSERT(*lock != NULL, "recursive lock allocation failed");
|
||||
|
||||
k_mutex_init((struct k_mutex *)*lock);
|
||||
}
|
||||
|
||||
/* Create a new dynamic non-recursive lock */
|
||||
void __retarget_lock_init(_LOCK_T *lock)
|
||||
{
|
||||
__retarget_lock_init_recursive(lock);
|
||||
}
|
||||
|
||||
/* Close dynamic recursive lock */
|
||||
void __retarget_lock_close_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
#ifndef CONFIG_USERSPACE
|
||||
free(lock);
|
||||
#else
|
||||
k_object_release(lock);
|
||||
#endif /* !CONFIG_USERSPACE */
|
||||
}
|
||||
|
||||
/* Close dynamic non-recursive lock */
|
||||
void __retarget_lock_close(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_close_recursive(lock);
|
||||
}
|
||||
|
||||
/* Acquiure recursive lock */
|
||||
void __retarget_lock_acquire_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
k_mutex_lock((struct k_mutex *)lock, K_FOREVER);
|
||||
}
|
||||
|
||||
/* Acquiure non-recursive lock */
|
||||
void __retarget_lock_acquire(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_acquire_recursive(lock);
|
||||
}
|
||||
|
||||
/* Try acquiring recursive lock */
|
||||
int __retarget_lock_try_acquire_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
return !k_mutex_lock((struct k_mutex *)lock, K_NO_WAIT);
|
||||
}
|
||||
|
||||
/* Try acquiring non-recursive lock */
|
||||
int __retarget_lock_try_acquire(_LOCK_T lock)
|
||||
{
|
||||
return __retarget_lock_try_acquire_recursive(lock);
|
||||
}
|
||||
|
||||
/* Release recursive lock */
|
||||
void __retarget_lock_release_recursive(_LOCK_T lock)
|
||||
{
|
||||
__ASSERT_NO_MSG(lock != NULL);
|
||||
k_mutex_unlock((struct k_mutex *)lock);
|
||||
}
|
||||
|
||||
/* Release non-recursive lock */
|
||||
void __retarget_lock_release(_LOCK_T lock)
|
||||
{
|
||||
__retarget_lock_release_recursive(lock);
|
||||
}
|
||||
|
||||
#endif /* CONFIG_MULTITHREADING */
|
|
@ -0,0 +1,32 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#ifndef _PICOLIBC_HOOKS_H_
|
||||
#define _PICOLIBC_HOOKS_H_
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <zephyr/posix/sys/stat.h>
|
||||
#include <sys/time.h>
|
||||
#include <zephyr/arch/cpu.h>
|
||||
#include <zephyr/linker/linker-defs.h>
|
||||
#include <zephyr/sys/util.h>
|
||||
#include <zephyr/sys/errno_private.h>
|
||||
#include <zephyr/sys/libc-hooks.h>
|
||||
#include <zephyr/internal/syscall_handler.h>
|
||||
#include <zephyr/app_memory/app_memdomain.h>
|
||||
#include <zephyr/init.h>
|
||||
#include <zephyr/sys/sem.h>
|
||||
#include <zephyr/logging/log.h>
|
||||
#ifdef CONFIG_MMU
|
||||
#include <zephyr/kernel/mm.h>
|
||||
#endif
|
||||
|
||||
#define LIBC_BSS K_APP_BMEM(z_libc_partition)
|
||||
#define LIBC_DATA K_APP_DMEM(z_libc_partition)
|
||||
|
||||
#endif /* _PICOLIBC_HOOKS_H_ */
|
|
@ -0,0 +1,54 @@
|
|||
/*
|
||||
* Copyright © 2021, Keith Packard <keithp@keithp.com>
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include "picolibc-hooks.h"
|
||||
|
||||
static LIBC_DATA int (*_stdout_hook)(int);
|
||||
|
||||
int z_impl_zephyr_fputc(int a, FILE *out)
|
||||
{
|
||||
(*_stdout_hook)(a);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_USERSPACE
|
||||
static inline int z_vrfy_zephyr_fputc(int c, FILE *stream)
|
||||
{
|
||||
return z_impl_zephyr_fputc(c, stream);
|
||||
}
|
||||
#include <zephyr/syscalls/zephyr_fputc_mrsh.c>
|
||||
#endif
|
||||
|
||||
static int picolibc_put(char a, FILE *f)
|
||||
{
|
||||
zephyr_fputc(a, f);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static LIBC_DATA FILE __stdout = FDEV_SETUP_STREAM(picolibc_put, NULL, NULL, 0);
|
||||
static LIBC_DATA FILE __stdin = FDEV_SETUP_STREAM(NULL, NULL, NULL, 0);
|
||||
|
||||
#ifdef __strong_reference
|
||||
#define STDIO_ALIAS(x) __strong_reference(stdout, x);
|
||||
#else
|
||||
#define STDIO_ALIAS(x) FILE *const x = &__stdout;
|
||||
#endif
|
||||
|
||||
FILE *const stdin = &__stdin;
|
||||
FILE *const stdout = &__stdout;
|
||||
STDIO_ALIAS(stderr);
|
||||
|
||||
void __stdout_hook_install(int (*hook)(int))
|
||||
{
|
||||
_stdout_hook = hook;
|
||||
__stdout.flags |= _FDEV_SETUP_WRITE;
|
||||
}
|
||||
|
||||
void __stdin_hook_install(unsigned char (*hook)(void))
|
||||
{
|
||||
__stdin.get = (int (*)(FILE *)) hook;
|
||||
__stdin.flags |= _FDEV_SETUP_READ;
|
||||
}
|
Loading…
Reference in New Issue