summaryrefslogtreecommitdiff
path: root/linux/src/include/asm-i386/semaphore.h
blob: 18e12c100c6baf86cf71f9f79cceaab0d4494e02 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
#ifndef _I386_SEMAPHORE_H
#define _I386_SEMAPHORE_H

#include <linux/linkage.h>
#include <asm/system.h>

/*
 * SMP- and interrupt-safe semaphores..
 *
 * (C) Copyright 1996 Linus Torvalds
 *
 * Modified 1996-12-23 by Dave Grothe <dave@gcom.com> to fix bugs in
 *                     the original code and to make semaphore waits
 *                     interruptible so that processes waiting on
 *                     semaphores can be killed.
 *
 * If you would like to see an analysis of this implementation, please
 * ftp to gcom.com and download the file
 * /pub/linux/src/semaphore/semaphore-2.0.24.tar.gz.
 *
 */

struct semaphore {
	int count;
	int waking;
	int lock ;			/* to make waking testing atomic */
	struct wait_queue * wait;
};

#define MUTEX ((struct semaphore) { 1, 0, 0, NULL })
#define MUTEX_LOCKED ((struct semaphore) { 0, 0, 0, NULL })

/* Special register calling convention:
 * eax contains return address
 * ecx contains semaphore address
 */
asmlinkage void down_failed(void /* special register calling convention */);
asmlinkage void up_wakeup(void /* special register calling convention */);

extern void __down(struct semaphore * sem);
extern void __up(struct semaphore * sem);

/*
 * This is ugly, but we want the default case to fall through.
 * "down_failed" is a special asm handler that calls the C
 * routine that actually waits. See arch/i386/lib/semaphore.S
 */
static inline void down(struct semaphore * sem)
{
	int d0;
	__asm__ __volatile__(
		"# atomic down operation\n\t"
		"movl $1f,%%eax\n\t"
#ifdef __SMP__
		"lock ; "
#endif
		"decl %1\n\t"
		"js " SYMBOL_NAME_STR(down_failed) "\n"
		"1:\n"
		:"=&a" (d0), "=m" (sem->count)
		:"c" (sem)
		:"memory");
}

/*
 * Primitives to spin on a lock.  Needed only for SMP version.
 */
extern inline void get_buzz_lock(int *lock_ptr)
{
#ifdef __SMP__
        while (xchg(lock_ptr,1) != 0) ;
#endif
} /* get_buzz_lock */

extern inline void give_buzz_lock(int *lock_ptr)
{
#ifdef __SMP__
        *lock_ptr = 0 ;
#endif
} /* give_buzz_lock */

asmlinkage int down_failed_interruptible(void);  /* params in registers */

/*
 * This version waits in interruptible state so that the waiting
 * process can be killed.  The down_failed_interruptible routine
 * returns negative for signalled and zero for semaphore acquired.
 */
static inline int down_interruptible(struct semaphore * sem)
{
	int	ret ;

        __asm__ __volatile__(
                "# atomic interruptible down operation\n\t"
                "movl $2f,%%eax\n\t"
#ifdef __SMP__
                "lock ; "
#endif
                "decl %1\n\t"
                "js " SYMBOL_NAME_STR(down_failed_interruptible) "\n\t"
                "xorl %%eax,%%eax\n"
                "2:\n"
                :"=&a" (ret), "=m" (sem->count)
                :"c" (sem)
                :"memory");

	return(ret) ;
}

/*
 * Note! This is subtle. We jump to wake people up only if
 * the semaphore was negative (== somebody was waiting on it).
 * The default case (no contention) will result in NO
 * jumps for both down() and up().
 */
static inline void up(struct semaphore * sem)
{
	int d0;
	__asm__ __volatile__(
		"# atomic up operation\n\t"
		"movl $1f,%%eax\n\t"
#ifdef __SMP__
		"lock ; "
#endif
		"incl %1\n\t"
		"jle " SYMBOL_NAME_STR(up_wakeup)
		"\n1:"
		:"=&a" (d0), "=m" (sem->count)
		:"c" (sem)
		:"memory");
}

#endif