Commit e9d9db6b authored by Linus Torvalds's avatar Linus Torvalds

Merge master.kernel.org:BK/linux-2.5

into home.transmeta.com:/home/torvalds/v2.5/linux
parents 8875527c 4fe9df79
...@@ -69,24 +69,27 @@ IF_MASK = 0x00000200 ...@@ -69,24 +69,27 @@ IF_MASK = 0x00000200
NT_MASK = 0x00004000 NT_MASK = 0x00004000
VM_MASK = 0x00020000 VM_MASK = 0x00020000
/* These are offsets into the irq_stat structure /*
* These are offsets into the irq_stat structure
* There is one per cpu and it is aligned to 32 * There is one per cpu and it is aligned to 32
* byte boundry (we put that here as a shift count) * byte boundry (we put that here as a shift count)
*/ */
irq_array_shift = CONFIG_X86_L1_CACHE_SHIFT irq_array_shift = CONFIG_X86_L1_CACHE_SHIFT
irq_stat_local_irq_count = 4 local_irq_count = 4
irq_stat_local_bh_count = 8 local_bh_count = 8
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
#define GET_CPU_INDX movl TI_CPU(%ebx),%eax; \ #define GET_CPU_IDX \
shll $irq_array_shift,%eax movl TI_CPU(%ebx), %eax; \
#define GET_CURRENT_CPU_INDX GET_THREAD_INFO(%ebx); \ shll $irq_array_shift, %eax
GET_CPU_INDX #define GET_CURRENT_CPU_IDX \
#define CPU_INDX (,%eax) GET_THREAD_INFO(%ebx); \
GET_CPU_IDX
#define CPU_IDX (,%eax)
#else #else
#define GET_CPU_INDX #define GET_CPU_IDX
#define GET_CURRENT_CPU_INDX GET_THREAD_INFO(%ebx) #define GET_CURRENT_CPU_IDX GET_THREAD_INFO(%ebx)
#define CPU_INDX #define CPU_IDX
#endif #endif
#ifdef CONFIG_PREEMPT #ifdef CONFIG_PREEMPT
...@@ -111,9 +114,9 @@ irq_stat_local_bh_count = 8 ...@@ -111,9 +114,9 @@ irq_stat_local_bh_count = 8
pushl %edx; \ pushl %edx; \
pushl %ecx; \ pushl %ecx; \
pushl %ebx; \ pushl %ebx; \
movl $(__KERNEL_DS),%edx; \ movl $(__KERNEL_DS), %edx; \
movl %edx,%ds; \ movl %edx, %ds; \
movl %edx,%es; movl %edx, %es;
#define RESTORE_ALL \ #define RESTORE_ALL \
popl %ebx; \ popl %ebx; \
...@@ -125,7 +128,7 @@ irq_stat_local_bh_count = 8 ...@@ -125,7 +128,7 @@ irq_stat_local_bh_count = 8
popl %eax; \ popl %eax; \
1: popl %ds; \ 1: popl %ds; \
2: popl %es; \ 2: popl %es; \
addl $4,%esp; \ addl $4, %esp; \
3: iret; \ 3: iret; \
.section .fixup,"ax"; \ .section .fixup,"ax"; \
4: movl $0,(%esp); \ 4: movl $0,(%esp); \
...@@ -147,20 +150,21 @@ irq_stat_local_bh_count = 8 ...@@ -147,20 +150,21 @@ irq_stat_local_bh_count = 8
.previous .previous
ENTRY(lcall7) ENTRY(lcall7)
pushfl # We get a different stack layout with call gates, pushfl # We get a different stack layout with call
pushl %eax # which has to be cleaned up later.. # gates, which has to be cleaned up later..
pushl %eax
SAVE_ALL SAVE_ALL
movl EIP(%esp),%eax # due to call gates, this is eflags, not eip.. movl EIP(%esp), %eax # due to call gates, this is eflags, not eip..
movl CS(%esp),%edx # this is eip.. movl CS(%esp), %edx # this is eip..
movl EFLAGS(%esp),%ecx # and this is cs.. movl EFLAGS(%esp), %ecx # and this is cs..
movl %eax,EFLAGS(%esp) # movl %eax,EFLAGS(%esp) #
movl %edx,EIP(%esp) # Now we move them to their "normal" places movl %edx,EIP(%esp) # Now we move them to their "normal" places
movl %ecx,CS(%esp) # movl %ecx,CS(%esp) #
movl %esp,%ebx movl %esp, %ebx
pushl %ebx pushl %ebx
andl $-8192,%ebx # GET_THREAD_INFO andl $-8192, %ebx # GET_THREAD_INFO
movl TI_EXEC_DOMAIN(%ebx),%edx # Get the execution domain movl TI_EXEC_DOMAIN(%ebx), %edx # Get the execution domain
movl 4(%edx),%edx # Get the lcall7 handler for the domain movl 4(%edx), %edx # Get the lcall7 handler for the domain
pushl $0x7 pushl $0x7
call *%edx call *%edx
addl $4, %esp addl $4, %esp
...@@ -168,20 +172,21 @@ ENTRY(lcall7) ...@@ -168,20 +172,21 @@ ENTRY(lcall7)
jmp resume_userspace jmp resume_userspace
ENTRY(lcall27) ENTRY(lcall27)
pushfl # We get a different stack layout with call gates, pushfl # We get a different stack layout with call
pushl %eax # which has to be cleaned up later.. # gates, which has to be cleaned up later..
pushl %eax
SAVE_ALL SAVE_ALL
movl EIP(%esp),%eax # due to call gates, this is eflags, not eip.. movl EIP(%esp), %eax # due to call gates, this is eflags, not eip..
movl CS(%esp),%edx # this is eip.. movl CS(%esp), %edx # this is eip..
movl EFLAGS(%esp),%ecx # and this is cs.. movl EFLAGS(%esp), %ecx # and this is cs..
movl %eax,EFLAGS(%esp) # movl %eax,EFLAGS(%esp) #
movl %edx,EIP(%esp) # Now we move them to their "normal" places movl %edx,EIP(%esp) # Now we move them to their "normal" places
movl %ecx,CS(%esp) # movl %ecx,CS(%esp) #
movl %esp,%ebx movl %esp, %ebx
pushl %ebx pushl %ebx
andl $-8192,%ebx # GET_THREAD_INFO andl $-8192, %ebx # GET_THREAD_INFO
movl TI_EXEC_DOMAIN(%ebx),%edx # Get the execution domain movl TI_EXEC_DOMAIN(%ebx), %edx # Get the execution domain
movl 4(%edx),%edx # Get the lcall7 handler for the domain movl 4(%edx), %edx # Get the lcall7 handler for the domain
pushl $0x27 pushl $0x27
call *%edx call *%edx
addl $4, %esp addl $4, %esp
...@@ -211,15 +216,17 @@ ENTRY(ret_from_intr) ...@@ -211,15 +216,17 @@ ENTRY(ret_from_intr)
GET_THREAD_INFO(%ebx) GET_THREAD_INFO(%ebx)
init_ret_intr init_ret_intr
ret_from_exception: ret_from_exception:
movl EFLAGS(%esp),%eax # mix EFLAGS and CS movl EFLAGS(%esp), %eax # mix EFLAGS and CS
movb CS(%esp),%al movb CS(%esp), %al
testl $(VM_MASK | 3),%eax testl $(VM_MASK | 3), %eax
jz resume_kernel # returning to kernel or vm86-space jz resume_kernel # returning to kernel or vm86-space
ENTRY(resume_userspace) ENTRY(resume_userspace)
cli # make sure we don't miss an interrupt setting need_resched cli # make sure we don't miss an interrupt
# or sigpending between sampling and the iret # setting need_resched or sigpending
movl TI_FLAGS(%ebx),%ecx # between sampling and the iret
andl $_TIF_WORK_MASK,%ecx # is there any work to be done on int/excp return? movl TI_FLAGS(%ebx), %ecx
andl $_TIF_WORK_MASK, %ecx # is there any work to be done on
# int/exception return?
jne work_pending jne work_pending
jmp restore_all jmp restore_all
...@@ -227,15 +234,17 @@ ENTRY(resume_userspace) ...@@ -227,15 +234,17 @@ ENTRY(resume_userspace)
ENTRY(resume_kernel) ENTRY(resume_kernel)
cmpl $0,TI_PRE_COUNT(%ebx) cmpl $0,TI_PRE_COUNT(%ebx)
jnz restore_all jnz restore_all
movl TI_FLAGS(%ebx),%ecx movl TI_FLAGS(%ebx), %ecx
testb $_TIF_NEED_RESCHED,%cl testb $_TIF_NEED_RESCHED, %cl
jz restore_all jz restore_all
movl SYMBOL_NAME(irq_stat)+irq_stat_local_bh_count CPU_INDX,%ecx movl SYMBOL_NAME(irq_stat)+local_bh_count CPU_IDX, %ecx
addl SYMBOL_NAME(irq_stat)+irq_stat_local_irq_count CPU_INDX,%ecx addl SYMBOL_NAME(irq_stat)+local_irq_count CPU_IDX, %ecx
jnz restore_all jnz restore_all
incl TI_PRE_COUNT(%ebx) incl TI_PRE_COUNT(%ebx)
sti sti
call SYMBOL_NAME(preempt_schedule) movl TI_TASK(%ebx), %ecx # ti->task
movl $0,(%ecx) # current->state = TASK_RUNNING
call SYMBOL_NAME(schedule)
jmp ret_from_intr jmp ret_from_intr
#endif #endif
...@@ -245,18 +254,20 @@ ENTRY(system_call) ...@@ -245,18 +254,20 @@ ENTRY(system_call)
pushl %eax # save orig_eax pushl %eax # save orig_eax
SAVE_ALL SAVE_ALL
GET_THREAD_INFO(%ebx) GET_THREAD_INFO(%ebx)
cmpl $(NR_syscalls),%eax cmpl $(NR_syscalls), %eax
jae syscall_badsys jae syscall_badsys
testb $_TIF_SYSCALL_TRACE,TI_FLAGS(%ebx) # system call tracing in operation # system call tracing in operation
testb $_TIF_SYSCALL_TRACE,TI_FLAGS(%ebx)
jnz syscall_trace_entry jnz syscall_trace_entry
syscall_call: syscall_call:
call *SYMBOL_NAME(sys_call_table)(,%eax,4) call *SYMBOL_NAME(sys_call_table)(,%eax,4)
movl %eax,EAX(%esp) # store the return value movl %eax,EAX(%esp) # store the return value
syscall_exit: syscall_exit:
cli # make sure we don't miss an interrupt setting need_resched cli # make sure we don't miss an interrupt
# or sigpending between sampling and the iret # setting need_resched or sigpending
movl TI_FLAGS(%ebx),%ecx # between sampling and the iret
testw $_TIF_ALLWORK_MASK,%cx # current->work movl TI_FLAGS(%ebx), %ecx
testw $_TIF_ALLWORK_MASK, %cx # current->work
jne syscall_exit_work jne syscall_exit_work
restore_all: restore_all:
RESTORE_ALL RESTORE_ALL
...@@ -264,23 +275,27 @@ restore_all: ...@@ -264,23 +275,27 @@ restore_all:
# perform work that needs to be done immediately before resumption # perform work that needs to be done immediately before resumption
ALIGN ALIGN
work_pending: work_pending:
testb $_TIF_NEED_RESCHED,%cl testb $_TIF_NEED_RESCHED, %cl
jz work_notifysig jz work_notifysig
work_resched: work_resched:
call SYMBOL_NAME(schedule) call SYMBOL_NAME(schedule)
cli # make sure we don't miss an interrupt setting need_resched cli # make sure we don't miss an interrupt
# or sigpending between sampling and the iret # setting need_resched or sigpending
movl TI_FLAGS(%ebx),%ecx # between sampling and the iret
andl $_TIF_WORK_MASK,%ecx # is there any work to be done other than syscall tracing? movl TI_FLAGS(%ebx), %ecx
andl $_TIF_WORK_MASK, %ecx # is there any work to be done other
# than syscall tracing?
jz restore_all jz restore_all
testb $_TIF_NEED_RESCHED,%cl testb $_TIF_NEED_RESCHED, %cl
jnz work_resched jnz work_resched
work_notifysig: # deal with pending signals and notify-resume requests work_notifysig: # deal with pending signals and
# notify-resume requests
testl $(VM_MASK),EFLAGS(%esp) testl $(VM_MASK),EFLAGS(%esp)
movl %esp,%eax movl %esp, %eax
jne work_notifysig_v86 # returning to kernel-space or vm86-space jne work_notifysig_v86 # returning to kernel-space or
xorl %edx,%edx # vm86-space
xorl %edx, %edx
call SYMBOL_NAME(do_notify_resume) call SYMBOL_NAME(do_notify_resume)
jmp restore_all jmp restore_all
...@@ -289,8 +304,8 @@ work_notifysig_v86: ...@@ -289,8 +304,8 @@ work_notifysig_v86:
pushl %ecx pushl %ecx
call SYMBOL_NAME(save_v86_state) call SYMBOL_NAME(save_v86_state)
popl %ecx popl %ecx
movl %eax,%esp movl %eax, %esp
xorl %edx,%edx xorl %edx, %edx
call SYMBOL_NAME(do_notify_resume) call SYMBOL_NAME(do_notify_resume)
jmp restore_all jmp restore_all
...@@ -298,22 +313,23 @@ work_notifysig_v86: ...@@ -298,22 +313,23 @@ work_notifysig_v86:
ALIGN ALIGN
syscall_trace_entry: syscall_trace_entry:
movl $-ENOSYS,EAX(%esp) movl $-ENOSYS,EAX(%esp)
movl %esp,%eax movl %esp, %eax
xorl %edx,%edx xorl %edx,%edx
call SYMBOL_NAME(do_syscall_trace) call SYMBOL_NAME(do_syscall_trace)
movl ORIG_EAX(%esp),%eax movl ORIG_EAX(%esp), %eax
cmpl $(NR_syscalls),%eax cmpl $(NR_syscalls), %eax
jnae syscall_call jnae syscall_call
jmp syscall_exit jmp syscall_exit
# perform syscall exit tracing # perform syscall exit tracing
ALIGN ALIGN
syscall_exit_work: syscall_exit_work:
testb $_TIF_SYSCALL_TRACE,%cl testb $_TIF_SYSCALL_TRACE, %cl
jz work_pending jz work_pending
sti # could let do_syscall_trace() call schedule() instead sti # could let do_syscall_trace() call
movl %esp,%eax # schedule() instead
movl $1,%edx movl %esp, %eax
movl $1, %edx
call SYMBOL_NAME(do_syscall_trace) call SYMBOL_NAME(do_syscall_trace)
jmp resume_userspace jmp resume_userspace
...@@ -329,7 +345,7 @@ ENTRY(divide_error) ...@@ -329,7 +345,7 @@ ENTRY(divide_error)
error_code: error_code:
pushl %ds pushl %ds
pushl %eax pushl %eax
xorl %eax,%eax xorl %eax, %eax
pushl %ebp pushl %ebp
pushl %edi pushl %edi
pushl %esi pushl %esi
...@@ -338,20 +354,20 @@ error_code: ...@@ -338,20 +354,20 @@ error_code:
pushl %ecx pushl %ecx
pushl %ebx pushl %ebx
cld cld
movl %es,%ecx movl %es, %ecx
movl ORIG_EAX(%esp), %esi # get the error code movl ORIG_EAX(%esp), %esi # get the error code
movl ES(%esp), %edi # get the function address movl ES(%esp), %edi # get the function address
movl %eax, ORIG_EAX(%esp) movl %eax, ORIG_EAX(%esp)
movl %ecx, ES(%esp) movl %ecx, ES(%esp)
movl %esp,%edx movl %esp, %edx
pushl %esi # push the error code pushl %esi # push the error code
pushl %edx # push the pt_regs pointer pushl %edx # push the pt_regs pointer
movl $(__KERNEL_DS),%edx movl $(__KERNEL_DS), %edx
movl %edx,%ds movl %edx, %ds
movl %edx,%es movl %edx, %es
GET_THREAD_INFO(%ebx) GET_THREAD_INFO(%ebx)
call *%edi call *%edi
addl $8,%esp addl $8, %esp
preempt_stop preempt_stop
jmp ret_from_exception jmp ret_from_exception
...@@ -369,8 +385,8 @@ ENTRY(device_not_available) ...@@ -369,8 +385,8 @@ ENTRY(device_not_available)
pushl $-1 # mark this as an int pushl $-1 # mark this as an int
SAVE_ALL SAVE_ALL
GET_THREAD_INFO(%ebx) GET_THREAD_INFO(%ebx)
movl %cr0,%eax movl %cr0, %eax
testl $0x4,%eax # EM (math emulation bit) testl $0x4, %eax # EM (math emulation bit)
jne device_not_available_emulate jne device_not_available_emulate
preempt_stop preempt_stop
call SYMBOL_NAME(math_state_restore) call SYMBOL_NAME(math_state_restore)
...@@ -378,7 +394,7 @@ ENTRY(device_not_available) ...@@ -378,7 +394,7 @@ ENTRY(device_not_available)
device_not_available_emulate: device_not_available_emulate:
pushl $0 # temporary storage for ORIG_EIP pushl $0 # temporary storage for ORIG_EIP
call SYMBOL_NAME(math_emulate) call SYMBOL_NAME(math_emulate)
addl $4,%esp addl $4, %esp
preempt_stop preempt_stop
jmp ret_from_exception jmp ret_from_exception
...@@ -390,11 +406,11 @@ ENTRY(debug) ...@@ -390,11 +406,11 @@ ENTRY(debug)
ENTRY(nmi) ENTRY(nmi)
pushl %eax pushl %eax
SAVE_ALL SAVE_ALL
movl %esp,%edx movl %esp, %edx
pushl $0 pushl $0
pushl %edx pushl %edx
call SYMBOL_NAME(do_nmi) call SYMBOL_NAME(do_nmi)
addl $8,%esp addl $8, %esp
RESTORE_ALL RESTORE_ALL
ENTRY(int3) ENTRY(int3)
...@@ -497,7 +513,8 @@ ENTRY(sys_call_table) ...@@ -497,7 +513,8 @@ ENTRY(sys_call_table)
.long SYMBOL_NAME(sys_ni_syscall) /* old gtty syscall holder */ .long SYMBOL_NAME(sys_ni_syscall) /* old gtty syscall holder */
.long SYMBOL_NAME(sys_access) .long SYMBOL_NAME(sys_access)
.long SYMBOL_NAME(sys_nice) .long SYMBOL_NAME(sys_nice)
.long SYMBOL_NAME(sys_ni_syscall) /* 35 */ /* old ftime syscall holder */ .long SYMBOL_NAME(sys_ni_syscall) /* 35 */
/* old ftime syscall holder */
.long SYMBOL_NAME(sys_sync) .long SYMBOL_NAME(sys_sync)
.long SYMBOL_NAME(sys_kill) .long SYMBOL_NAME(sys_kill)
.long SYMBOL_NAME(sys_rename) .long SYMBOL_NAME(sys_rename)
......
...@@ -455,9 +455,11 @@ void do_syscall_trace(struct pt_regs *regs, int entryexit) ...@@ -455,9 +455,11 @@ void do_syscall_trace(struct pt_regs *regs, int entryexit)
between a syscall stop and SIGTRAP delivery */ between a syscall stop and SIGTRAP delivery */
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0);
preempt_disable();
current->state = TASK_STOPPED; current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD); notify_parent(current, SIGCHLD);
schedule(); schedule();
preempt_enable();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -610,9 +610,11 @@ int do_signal(struct pt_regs *regs, sigset_t *oldset) ...@@ -610,9 +610,11 @@ int do_signal(struct pt_regs *regs, sigset_t *oldset)
if ((current->ptrace & PT_PTRACED) && signr != SIGKILL) { if ((current->ptrace & PT_PTRACED) && signr != SIGKILL) {
/* Let the debugger run. */ /* Let the debugger run. */
current->exit_code = signr; current->exit_code = signr;
preempt_disable();
current->state = TASK_STOPPED; current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD); notify_parent(current, SIGCHLD);
schedule(); schedule();
preempt_enable();
/* We're back. Did the debugger cancel the sig? */ /* We're back. Did the debugger cancel the sig? */
if (!(signr = current->exit_code)) if (!(signr = current->exit_code))
...@@ -667,12 +669,14 @@ int do_signal(struct pt_regs *regs, sigset_t *oldset) ...@@ -667,12 +669,14 @@ int do_signal(struct pt_regs *regs, sigset_t *oldset)
case SIGSTOP: { case SIGSTOP: {
struct signal_struct *sig; struct signal_struct *sig;
current->state = TASK_STOPPED;
current->exit_code = signr; current->exit_code = signr;
sig = current->p_pptr->sig; sig = current->p_pptr->sig;
preempt_disable();
current->state = TASK_STOPPED;
if (sig && !(sig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDSTOP)) if (sig && !(sig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDSTOP))
notify_parent(current, SIGCHLD); notify_parent(current, SIGCHLD);
schedule(); schedule();
preempt_enable();
continue; continue;
} }
......
...@@ -415,6 +415,25 @@ static __inline__ unsigned long __ffs(unsigned long word) ...@@ -415,6 +415,25 @@ static __inline__ unsigned long __ffs(unsigned long word)
#ifdef __KERNEL__ #ifdef __KERNEL__
/*
* Every architecture must define this function. It's the fastest
* way of searching a 140-bit bitmap where the first 100 bits are
* unlikely to be set. It's guaranteed that at least one of the 140
* bits is cleared.
*/
static inline int sched_find_first_bit(unsigned long *b)
{
if (unlikely(b[0]))
return __ffs(b[0]);
if (unlikely(b[1]))
return __ffs(b[1]) + 32;
if (unlikely(b[2]))
return __ffs(b[2]) + 64;
if (b[3])
return __ffs(b[3]) + 96;
return __ffs(b[4]) + 128;
}
/** /**
* ffs - find first bit set * ffs - find first bit set
* @x: the word to search * @x: the word to search
......
...@@ -6,24 +6,6 @@ ...@@ -6,24 +6,6 @@
#include <asm/atomic.h> #include <asm/atomic.h>
#include <asm/pgalloc.h> #include <asm/pgalloc.h>
/*
* Every architecture must define this function. It's the fastest
* way of searching a 140-bit bitmap where the first 100 bits are
* unlikely to be set. It's guaranteed that at least one of the 140
* bits is cleared.
*/
static inline int sched_find_first_bit(unsigned long *b)
{
if (unlikely(b[0]))
return __ffs(b[0]);
if (unlikely(b[1]))
return __ffs(b[1]) + 32;
if (unlikely(b[2]))
return __ffs(b[2]) + 64;
if (b[3])
return __ffs(b[3]) + 96;
return __ffs(b[4]) + 128;
}
/* /*
* possibly do the LDT unload here? * possibly do the LDT unload here?
*/ */
......
...@@ -13,24 +13,23 @@ struct task_struct; /* one of the stranger aspects of C forward declarations.. * ...@@ -13,24 +13,23 @@ struct task_struct; /* one of the stranger aspects of C forward declarations.. *
extern void FASTCALL(__switch_to(struct task_struct *prev, struct task_struct *next)); extern void FASTCALL(__switch_to(struct task_struct *prev, struct task_struct *next));
#define prepare_to_switch() do { } while(0) #define prepare_to_switch() do { } while(0)
#define switch_to(prev,next,last) do { \
#define switch_to(prev,next) do { \
asm volatile("pushl %%esi\n\t" \ asm volatile("pushl %%esi\n\t" \
"pushl %%edi\n\t" \ "pushl %%edi\n\t" \
"pushl %%ebp\n\t" \ "pushl %%ebp\n\t" \
"movl %%esp,%0\n\t" /* save ESP */ \ "movl %%esp,%0\n\t" /* save ESP */ \
"movl %3,%%esp\n\t" /* restore ESP */ \ "movl %2,%%esp\n\t" /* restore ESP */ \
"movl $1f,%1\n\t" /* save EIP */ \ "movl $1f,%1\n\t" /* save EIP */ \
"pushl %4\n\t" /* restore EIP */ \ "pushl %3\n\t" /* restore EIP */ \
"jmp __switch_to\n" \ "jmp __switch_to\n" \
"1:\t" \ "1:\t" \
"popl %%ebp\n\t" \ "popl %%ebp\n\t" \
"popl %%edi\n\t" \ "popl %%edi\n\t" \
"popl %%esi\n\t" \ "popl %%esi\n\t" \
:"=m" (prev->thread.esp),"=m" (prev->thread.eip), \ :"=m" (prev->thread.esp),"=m" (prev->thread.eip) \
"=b" (last) \
:"m" (next->thread.esp),"m" (next->thread.eip), \ :"m" (next->thread.esp),"m" (next->thread.eip), \
"a" (prev), "d" (next), \ "a" (prev), "d" (next)); \
"b" (prev)); \
} while (0) } while (0)
#define _set_base(addr,base) do { unsigned long __pr; \ #define _set_base(addr,base) do { unsigned long __pr; \
......
...@@ -92,7 +92,6 @@ extern unsigned long nr_running(void); ...@@ -92,7 +92,6 @@ extern unsigned long nr_running(void);
#define TASK_UNINTERRUPTIBLE 2 #define TASK_UNINTERRUPTIBLE 2
#define TASK_ZOMBIE 4 #define TASK_ZOMBIE 4
#define TASK_STOPPED 8 #define TASK_STOPPED 8
#define PREEMPT_ACTIVE 0x4000000
#define __set_task_state(tsk, state_value) \ #define __set_task_state(tsk, state_value) \
do { (tsk)->state = (state_value); } while (0) do { (tsk)->state = (state_value); } while (0)
......
...@@ -177,8 +177,7 @@ do { \ ...@@ -177,8 +177,7 @@ do { \
do { \ do { \
--current_thread_info()->preempt_count; \ --current_thread_info()->preempt_count; \
barrier(); \ barrier(); \
if (unlikely(!(current_thread_info()->preempt_count) && \ if (unlikely(test_thread_flag(TIF_NEED_RESCHED))) \
test_thread_flag(TIF_NEED_RESCHED))) \
preempt_schedule(); \ preempt_schedule(); \
} while (0) } while (0)
......
...@@ -476,7 +476,12 @@ static void exit_notify(void) ...@@ -476,7 +476,12 @@ static void exit_notify(void)
write_lock_irq(&tasklist_lock); write_lock_irq(&tasklist_lock);
} }
} }
write_unlock_irq(&tasklist_lock); /*
* No need to unlock IRQs, we'll schedule() immediately
* anyway. In the preemption case this also makes it
* impossible for the task to get runnable again.
*/
write_unlock(&tasklist_lock);
} }
NORET_TYPE void do_exit(long code) NORET_TYPE void do_exit(long code)
......
...@@ -435,17 +435,8 @@ static inline void context_switch(task_t *prev, task_t *next) ...@@ -435,17 +435,8 @@ static inline void context_switch(task_t *prev, task_t *next)
mmdrop(oldmm); mmdrop(oldmm);
} }
/* /* Here we just switch the register state and the stack. */
* Here we just switch the register state and the stack. There are switch_to(prev, next);
* 3 processes affected by a context switch:
*
* prev ==> .... ==> (last => next)
*
* It's the 'much more previous' 'prev' that is on next's stack,
* but prev is set to (the just run) 'last' process by switch_to().
* This might sound slightly confusing but makes tons of sense.
*/
switch_to(prev, next, prev);
} }
unsigned long nr_running(void) unsigned long nr_running(void)
...@@ -770,6 +761,7 @@ asmlinkage void schedule(void) ...@@ -770,6 +761,7 @@ asmlinkage void schedule(void)
if (unlikely(in_interrupt())) if (unlikely(in_interrupt()))
BUG(); BUG();
need_resched:
preempt_disable(); preempt_disable();
prev = current; prev = current;
rq = this_rq(); rq = this_rq();
...@@ -778,15 +770,6 @@ asmlinkage void schedule(void) ...@@ -778,15 +770,6 @@ asmlinkage void schedule(void)
prev->sleep_timestamp = jiffies; prev->sleep_timestamp = jiffies;
spin_lock_irq(&rq->lock); spin_lock_irq(&rq->lock);
#ifdef CONFIG_PREEMPT
/*
* if entering from preempt_schedule, off a kernel preemption,
* go straight to picking the next task.
*/
if (unlikely(preempt_get_count() & PREEMPT_ACTIVE))
goto pick_next_task;
#endif
switch (prev->state) { switch (prev->state) {
case TASK_INTERRUPTIBLE: case TASK_INTERRUPTIBLE:
if (unlikely(signal_pending(prev))) { if (unlikely(signal_pending(prev))) {
...@@ -798,7 +781,7 @@ asmlinkage void schedule(void) ...@@ -798,7 +781,7 @@ asmlinkage void schedule(void)
case TASK_RUNNING: case TASK_RUNNING:
; ;
} }
#if CONFIG_SMP || CONFIG_PREEMPT #if CONFIG_SMP
pick_next_task: pick_next_task:
#endif #endif
if (unlikely(!rq->nr_running)) { if (unlikely(!rq->nr_running)) {
...@@ -847,6 +830,8 @@ asmlinkage void schedule(void) ...@@ -847,6 +830,8 @@ asmlinkage void schedule(void)
reacquire_kernel_lock(current); reacquire_kernel_lock(current);
preempt_enable_no_resched(); preempt_enable_no_resched();
if (test_thread_flag(TIF_NEED_RESCHED))
goto need_resched;
return; return;
} }
...@@ -856,12 +841,10 @@ asmlinkage void schedule(void) ...@@ -856,12 +841,10 @@ asmlinkage void schedule(void)
*/ */
asmlinkage void preempt_schedule(void) asmlinkage void preempt_schedule(void)
{ {
do { if (unlikely(preempt_get_count()))
current_thread_info()->preempt_count += PREEMPT_ACTIVE; return;
current->state = TASK_RUNNING;
schedule(); schedule();
current_thread_info()->preempt_count -= PREEMPT_ACTIVE;
barrier();
} while (test_thread_flag(TIF_NEED_RESCHED));
} }
#endif /* CONFIG_PREEMPT */ #endif /* CONFIG_PREEMPT */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment