mirror of
https://git.musl-libc.org/git/musl
synced 2025-01-09 00:02:17 +03:00
implement flockfile api, rework stdio locking
This commit is contained in:
parent
d8d19f4d46
commit
5eb0d33ec0
@ -15,6 +15,9 @@ struct __libc {
|
|||||||
int (*rsyscall)(int, long, long, long, long, long, long);
|
int (*rsyscall)(int, long, long, long, long, long, long);
|
||||||
void (**tsd_keys)(void *);
|
void (**tsd_keys)(void *);
|
||||||
void (*fork_handler)(int);
|
void (*fork_handler)(int);
|
||||||
|
FILE *ofl_head;
|
||||||
|
int ofl_lock;
|
||||||
|
void (*lockfile)(FILE *);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -36,6 +39,7 @@ extern struct __libc *__libc_loc(void) __attribute__((const));
|
|||||||
|
|
||||||
/* Designed to avoid any overhead in non-threaded processes */
|
/* Designed to avoid any overhead in non-threaded processes */
|
||||||
void __lock(volatile int *);
|
void __lock(volatile int *);
|
||||||
|
void __lockfile(FILE *);
|
||||||
#define LOCK(x) (libc.threads_minus_1 ? (__lock(x),1) : ((void)(x),1))
|
#define LOCK(x) (libc.threads_minus_1 ? (__lock(x),1) : ((void)(x),1))
|
||||||
#define UNLOCK(x) (*(x)=0)
|
#define UNLOCK(x) (*(x)=0)
|
||||||
#define CANCELPT(x) (libc.cancelpt ? libc.cancelpt((x)),0 : (void)(x),0)
|
#define CANCELPT(x) (libc.cancelpt ? libc.cancelpt((x)),0 : (void)(x),0)
|
||||||
|
@ -23,8 +23,8 @@
|
|||||||
|
|
||||||
#define UNGET 4
|
#define UNGET 4
|
||||||
|
|
||||||
#define FLOCK(f) LOCK(&f->lock)
|
#define FLOCK(f) ((libc.lockfile && (f)->owner>=0) ? (libc.lockfile((f)),0) : 0)
|
||||||
#define FUNLOCK(f) UNLOCK(&f->lock)
|
#define FUNLOCK(f) ((f)->lockcount && (--(f)->lockcount || ((f)->owner=(f)->lock=0)))
|
||||||
|
|
||||||
#define F_PERM 1
|
#define F_PERM 1
|
||||||
#define F_NORD 4
|
#define F_NORD 4
|
||||||
@ -50,7 +50,7 @@ struct __FILE_s {
|
|||||||
signed char lbf;
|
signed char lbf;
|
||||||
int lock;
|
int lock;
|
||||||
int lockcount;
|
int lockcount;
|
||||||
void *owner;
|
void *dummy5;
|
||||||
off_t off;
|
off_t off;
|
||||||
int (*flush)(FILE *);
|
int (*flush)(FILE *);
|
||||||
void **wide_data; /* must be NULL */
|
void **wide_data; /* must be NULL */
|
||||||
@ -59,6 +59,7 @@ struct __FILE_s {
|
|||||||
off_t (*seek)(FILE *, off_t, int);
|
off_t (*seek)(FILE *, off_t, int);
|
||||||
int mode;
|
int mode;
|
||||||
int (*close)(FILE *);
|
int (*close)(FILE *);
|
||||||
|
int owner;
|
||||||
};
|
};
|
||||||
|
|
||||||
size_t __stdio_read(FILE *, unsigned char *, size_t);
|
size_t __stdio_read(FILE *, unsigned char *, size_t);
|
||||||
@ -80,15 +81,9 @@ int __putc_unlocked(int, FILE *);
|
|||||||
|
|
||||||
FILE *__fdopen(int, const char *);
|
FILE *__fdopen(int, const char *);
|
||||||
|
|
||||||
extern struct ofl
|
#define OFLLOCK() LOCK(&libc.ofl_lock)
|
||||||
{
|
#define OFLUNLOCK() UNLOCK(&libc.ofl_lock)
|
||||||
FILE *head;
|
#define ofl_head (libc.ofl_head)
|
||||||
int lock;
|
|
||||||
} __ofl;
|
|
||||||
|
|
||||||
#define OFLLOCK() LOCK(&__ofl.lock)
|
|
||||||
#define OFLUNLOCK() UNLOCK(&__ofl.lock)
|
|
||||||
#define ofl_head (__ofl.head)
|
|
||||||
|
|
||||||
#define feof(f) ((f)->flags & F_EOF)
|
#define feof(f) ((f)->flags & F_EOF)
|
||||||
#define ferror(f) ((f)->flags & F_ERR)
|
#define ferror(f) ((f)->flags & F_ERR)
|
||||||
|
19
src/stdio/__lockfile.c
Normal file
19
src/stdio/__lockfile.c
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
#include "stdio_impl.h"
|
||||||
|
#include "pthread_impl.h"
|
||||||
|
|
||||||
|
void __lockfile(FILE *f)
|
||||||
|
{
|
||||||
|
int spins;
|
||||||
|
if (f->owner < 0) return;
|
||||||
|
if (f->owner && f->owner == __pthread_self()->tid) {
|
||||||
|
while (f->lockcount == INT_MAX);
|
||||||
|
f->lockcount++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
spins = 100000;
|
||||||
|
while (a_swap(&f->lock, 1))
|
||||||
|
if (spins) spins--, a_spin();
|
||||||
|
else syscall0(__NR_sched_yield);
|
||||||
|
f->owner = __pthread_self()->tid;
|
||||||
|
f->lockcount = 1;
|
||||||
|
}
|
@ -1,3 +0,0 @@
|
|||||||
#include "stdio_impl.h"
|
|
||||||
|
|
||||||
struct ofl __ofl;
|
|
9
src/stdio/flockfile.c
Normal file
9
src/stdio/flockfile.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
#include "stdio_impl.h"
|
||||||
|
#include "pthread_impl.h"
|
||||||
|
|
||||||
|
void flockfile(FILE *f)
|
||||||
|
{
|
||||||
|
pthread_self();
|
||||||
|
libc.lockfile = __lockfile;
|
||||||
|
__lockfile(f);
|
||||||
|
}
|
18
src/stdio/ftrylockfile.c
Normal file
18
src/stdio/ftrylockfile.c
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
#include "stdio_impl.h"
|
||||||
|
#include "pthread_impl.h"
|
||||||
|
|
||||||
|
int ftrylockfile(FILE *f)
|
||||||
|
{
|
||||||
|
libc.lockfile = __lockfile;
|
||||||
|
if (f->owner && f->owner == pthread_self()->tid) {
|
||||||
|
if (f->lockcount == INT_MAX)
|
||||||
|
return -1;
|
||||||
|
f->lockcount++;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (a_swap(&f->lock, 1))
|
||||||
|
return -1;
|
||||||
|
f->owner = pthread_self()->tid;
|
||||||
|
f->lockcount = 1;
|
||||||
|
return 0;
|
||||||
|
}
|
7
src/stdio/funlockfile.c
Normal file
7
src/stdio/funlockfile.c
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#include "stdio_impl.h"
|
||||||
|
#include "pthread_impl.h"
|
||||||
|
|
||||||
|
void funlockfile(FILE *f)
|
||||||
|
{
|
||||||
|
FUNLOCK(f);
|
||||||
|
}
|
@ -17,6 +17,7 @@ int vsnprintf(char *s, size_t n, const char *fmt, va_list ap)
|
|||||||
f.write = sn_write;
|
f.write = sn_write;
|
||||||
f.buf_size = 1;
|
f.buf_size = 1;
|
||||||
f.buf = buf;
|
f.buf = buf;
|
||||||
|
f.owner = -1;
|
||||||
if (n > INT_MAX) {
|
if (n > INT_MAX) {
|
||||||
errno = EOVERFLOW;
|
errno = EOVERFLOW;
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -150,6 +150,7 @@ static void init_threads()
|
|||||||
{
|
{
|
||||||
struct sigaction sa = { .sa_flags = SA_SIGINFO | SA_RESTART };
|
struct sigaction sa = { .sa_flags = SA_SIGINFO | SA_RESTART };
|
||||||
libc.lock = __lock;
|
libc.lock = __lock;
|
||||||
|
libc.lockfile = __lockfile;
|
||||||
libc.cancelpt = cancelpt;
|
libc.cancelpt = cancelpt;
|
||||||
libc.rsyscall = rsyscall;
|
libc.rsyscall = rsyscall;
|
||||||
sa.sa_sigaction = cancel_handler;
|
sa.sa_sigaction = cancel_handler;
|
||||||
|
Loading…
Reference in New Issue
Block a user