- 论坛徽章:
- 0
|
hi,all
我碰到一个get_next_timer_interrupt的死机问题,objdump看到应该是timer_list为NULL,我猜可能是mod_timer后未del_timer_sync而kfree了导致出此问题。所以做实验在setup_timer,mod_timer后 和timer到来之前kfree,想重现此现象,但实际得到的错误是:kernel BUG at kernel/timer.c:1035!
那么该问题到底是什么造成的呢?现象不好重现。
有个网页显示driver未调用del_timer_sync也有出get_next_timer_interrupt问题,lk版本不同,死机函数offset不同,不确定是否是同一类问题
http://www.gossamer-threads.com/lists/linux/kernel/886925
lk3.0.8 ARM A9 dual core
- 00000b2c <get_next_timer_interrupt>:
- /**
- * get_next_timer_interrupt - return the jiffy of the next pending timer
- * @now: current time (in jiffies)
- */
- unsigned long get_next_timer_interrupt(unsigned long now)
- {
- b2c: e92d4ff0 push {r4, r5, r6, r7, r8, r9, sl, fp, lr}
- b30: e24dd024 sub sp, sp, #36 ; 0x24
- struct tvec_base *base = __this_cpu_read(tvec_bases);
- b34: e3006000 movw r6, #0
- b38: e3406000 movt r6, #0
- b3c: e1a0100d mov r1, sp
- b40: e3c15d7f bic r5, r1, #8128 ; 0x1fc0
- b44: e3c5203f bic r2, r5, #63 ; 0x3f
- /**
- * get_next_timer_interrupt - return the jiffy of the next pending timer
- * @now: current time (in jiffies)
- */
- unsigned long get_next_timer_interrupt(unsigned long now)
- {
- b48: e1a05000 mov r5, r0
- struct tvec_base *base = __this_cpu_read(tvec_bases);
- b4c: e300e000 movw lr, #0
- b50: e340e000 movt lr, #0
- b54: e592c014 ldr ip, [r2, #20]
- b58: e796010c ldr r0, [r6, ip, lsl #2]
- b5c: e79e6000 ldr r6, [lr, r0]
- /*
- * Pretend that there is no timer pending if the cpu is offline.
- * Possible pending timers will be migrated later to an active cpu.
- */
- if (cpu_is_offline(smp_processor_id()))
- b60: ebfffffe bl 0 <debug_smp_processor_id>
- * @nr: bit number to test
- * @addr: Address to start counting from
- */
- static inline int test_bit(int nr, const volatile unsigned long *addr)
- {
- return 1UL & (addr[BIT_WORD(nr)] >> (nr & (BITS_PER_LONG-1)));
- b64: e3003000 movw r3, #0
- b68: e3403000 movt r3, #0
- b6c: e593c000 ldr ip, [r3]
- b70: e3500000 cmp r0, #0
- b74: e280101f add r1, r0, #31
- b78: a1a01000 movge r1, r0
- b7c: e200001f and r0, r0, #31
- b80: e1a022c1 asr r2, r1, #5
- b84: e79c1102 ldr r1, [ip, r2, lsl #2]
- b88: e1a03031 lsr r3, r1, r0
- b8c: e3130001 tst r3, #1
- return now + NEXT_TIMER_MAX_DELTA;
- b90: 02454107 subeq r4, r5, #-1073741823 ; 0xc0000001
- /*
- * Pretend that there is no timer pending if the cpu is offline.
- * Possible pending timers will be migrated later to an active cpu.
- */
- if (cpu_is_offline(smp_processor_id()))
- b94: 0a000079 beq d80 <get_next_timer_interrupt+0x254>
- raw_spin_lock_init(&(_lock)->rlock); \
- } while (0)
- static inline void spin_lock(spinlock_t *lock)
- {
- raw_spin_lock(&lock->rlock);
- b98: e1a00006 mov r0, r6
- b9c: ebfffffe bl 0 <_raw_spin_lock>
- return now + NEXT_TIMER_MAX_DELTA;
- spin_lock(&base->lock);
- if (time_before_eq(base->next_timer, base->timer_jiffies))
- ba0: e596a00c ldr sl, [r6, #12]
- ba4: e5964010 ldr r4, [r6, #16]
- ba8: e064300a rsb r3, r4, sl
- bac: e3530000 cmp r3, #0
- bb0: ba00004e blt cf0 <get_next_timer_interrupt+0x1c4>
- int index, slot, array, found = 0;
- struct timer_list *nte;
- struct tvec *varray[4];
- /* Look for timer events in tv1. */
- index = slot = timer_jiffies & TVR_MASK;
- bb4: e20a20ff and r2, sl, #255 ; 0xff
- bb8: e1a01002 mov r1, r2
- do {
- list_for_each_entry(nte, base->tv1.vec + slot, entry) {
- bbc: e0864181 add r4, r6, r1, lsl #3
- bc0: e2840014 add r0, r4, #20
- bc4: e5943014 ldr r3, [r4, #20]
- bc8: e1530000 cmp r3, r0
- bcc: 0a000005 beq be8 <get_next_timer_interrupt+0xbc>
- if (tbase_get_deferrable(nte->base))
- bd0: e593e00c ldr lr, [r3, #12]
- bd4: e31e0001 tst lr, #1
- bd8: 0a00003a beq cc8 <get_next_timer_interrupt+0x19c>
- struct tvec *varray[4];
- /* Look for timer events in tv1. */
- index = slot = timer_jiffies & TVR_MASK;
- do {
- list_for_each_entry(nte, base->tv1.vec + slot, entry) {
- bdc: e5933000 ldr r3, [r3]
- be0: e1530000 cmp r3, r0
- be4: 1afffff9 bne bd0 <get_next_timer_interrupt+0xa4>
- /* Look at the cascade bucket(s)? */
- if (!index || slot < index)
- goto cascade;
- return expires;
- }
- slot = (slot + 1) & TVR_MASK;
- be8: e2810001 add r0, r1, #1
- bec: e20010ff and r1, r0, #255 ; 0xff
- } while (slot != index);
- bf0: e1520001 cmp r2, r1
- bf4: 1afffff0 bne bbc <get_next_timer_interrupt+0x90>
- * This function needs to be called with interrupts disabled.
- */
- static unsigned long __next_timer_interrupt(struct tvec_base *base)
- {
- unsigned long timer_jiffies = base->timer_jiffies;
- unsigned long expires = timer_jiffies + NEXT_TIMER_MAX_DELTA;
- bf8: e24a4107 sub r4, sl, #-1073741823 ; 0xc0000001
- bfc: e3a00000 mov r0, #0
- slot = (slot + 1) & TVR_MASK;
- } while (slot != index);
- cascade:
- /* Calculate the next cascade event */
- if (index)
- c00: e3520000 cmp r2, #0
- slot = (slot + 1) & TVN_MASK;
- } while (slot != index);
- if (index)
- timer_jiffies += TVN_SIZE - index;
- timer_jiffies >>= TVN_BITS;
- c04: e28d9010 add r9, sp, #16
- } while (slot != index);
- cascade:
- /* Calculate the next cascade event */
- if (index)
- timer_jiffies += TVR_SIZE - index;
- c08: 128aac01 addne sl, sl, #256 ; 0x100
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- varray[1] = &base->tv3;
- c0c: e2867ea1 add r7, r6, #2576 ; 0xa10
- } while (slot != index);
- cascade:
- /* Calculate the next cascade event */
- if (index)
- timer_jiffies += TVR_SIZE - index;
- c10: 1062a00a rsbne sl, r2, sl
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- c14: e286ee81 add lr, r6, #2064 ; 0x810
- varray[1] = &base->tv3;
- varray[2] = &base->tv4;
- c18: e286cec1 add ip, r6, #3088 ; 0xc10
- varray[3] = &base->tv5;
- c1c: e2863ee1 add r3, r6, #3600 ; 0xe10
- cascade:
- /* Calculate the next cascade event */
- if (index)
- timer_jiffies += TVR_SIZE - index;
- timer_jiffies >>= TVR_BITS;
- c20: e1a0a42a lsr sl, sl, #8
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- varray[1] = &base->tv3;
- varray[2] = &base->tv4;
- varray[3] = &base->tv5;
- c24: e3a08000 mov r8, #0
- timer_jiffies += TVR_SIZE - index;
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- varray[1] = &base->tv3;
- c28: e2871004 add r1, r7, #4
- if (index)
- timer_jiffies += TVR_SIZE - index;
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- c2c: e28ee004 add lr, lr, #4
- varray[1] = &base->tv3;
- varray[2] = &base->tv4;
- c30: e28c2004 add r2, ip, #4
- varray[3] = &base->tv5;
- c34: e2837004 add r7, r3, #4
- timer_jiffies += TVR_SIZE - index;
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- varray[1] = &base->tv3;
- c38: e58d1014 str r1, [sp, #20]
- varray[2] = &base->tv4;
- c3c: e58d2018 str r2, [sp, #24]
- varray[3] = &base->tv5;
- c40: e58d701c str r7, [sp, #28]
- if (index)
- timer_jiffies += TVR_SIZE - index;
- timer_jiffies >>= TVR_BITS;
- /* Check tv2-tv5. */
- varray[0] = &base->tv2;
- c44: e58de010 str lr, [sp, #16]
- varray[3] = &base->tv5;
- for (array = 0; array < 4; array++) {
- struct tvec *varp = varray[array];
- index = slot = timer_jiffies & TVN_MASK;
- c48: e20a703f and r7, sl, #63 ; 0x3f
- c4c: e1a0c007 mov ip, r7
- do {
- list_for_each_entry(nte, varp->vec + slot, entry) {
- c50: e79e318c ldr r3, [lr, ip, lsl #3]
- c54: e08e118c add r1, lr, ip, lsl #3
- c58: e1530001 cmp r3, r1
- c5c: 0a00000a beq c8c <get_next_timer_interrupt+0x160>
- if (tbase_get_deferrable(nte->base))
- c60: e593200c ldr r2, [r3, #12]
- c64: e3120001 tst r2, #1
- c68: 1a000004 bne c80 <get_next_timer_interrupt+0x154>
- continue;
- found = 1;
- if (time_before(nte->expires, expires))
- c6c: e5932008 ldr r2, [r3, #8]
- c70: e3a00001 mov r0, #1
- c74: e064b002 rsb fp, r4, r2
- c78: e35b0000 cmp fp, #0
- c7c: b1a04002 movlt r4, r2
复制代码 |
|