LKML Archive on lore.kernel.org
help / color / mirror / Atom feed
* Strange oops in iret_exc with my own module
@ 2007-02-09  2:26 Grzegorz Kulewski
  0 siblings, 0 replies; only message in thread
From: Grzegorz Kulewski @ 2007-02-09  2:26 UTC (permalink / raw)
  To: linux-kernel

Hi,

I written really simple and stupid module (the design wasn't mine). One 
tester of this module provided me the following oops it caused quite often 
(on many automated tests):

[17179765.492000] BUG: unable to handle kernel NULL pointer dereference at 
virtual address 00000000
[17179765.492000]  printing eip:
[17179765.492000] c02c4845
[17179765.492000] *pde = 00000000
[17179765.492000] Oops: 0002 [#1]
[17179765.492000] Modules linked in: altpinfo rfcomm l2cap bluetooth ppdev 
cpufreq_userspace cpufreq_stats freq_table
cpufreq_powersave cpufreq_ondemand cpufreq_conservative video button 
battery container ac af_packet dm_mod md_mod ipv6
lp snd_seq_dummy snd_seq_oss snd_seq_midi snd_seq_midi_event snd_seq 
snd_via82xx gameport snd_ac97_codec snd_ac97_bus
via686a snd_pcm_oss snd_mixer_oss tsdev i2c_isa i2c_viapro usbhid snd_pcm 
snd_timer snd_page_alloc snd_mpu401_uart
snd_rawmidi snd_seq_device i2c_core snd soundcore psmouse 8139too mii 
pcspkr floppy serio_raw parport_pc rtc via_agp
agpgart parport shpchp pci_hotplug evdev ext3 jbd mbcache ide_generic 
uhci_hcd usbcore ide_cd cdrom ide_disk via82cxxx
generic thermal processor fan vga16fb cfbcopyarea vgastate cfbimgblt 
cfbfillrect
[17179765.492000] CPU:    0
[17179765.492000] EIP:    0060:[<c02c4845>]    Not tainted VLI
[17179765.492000] EFLAGS: 00210246   (2.6.17.13nopreempt #3)
[17179765.492000] EIP is at iret_exc+0x4fd/0x75d
[17179765.492000] eax: 00000000   ebx: 00000000   ecx: 2b646970   edx: 
336e3094
[17179765.492000] esi: 0809c724   edi: 00000000   ebp: 00000003   esp: 
c32b3f34
[17179765.492000] ds: 007b   es: 007b   ss: 0068
[17179765.492000] Process so6-3-test (pid: 7227, threadinfo=c32b2000 
task=c620e550)
[17179765.492000] Stack: 00000000 2b646970 00000000 2b646970 d0833913 
00000000 0809c724 2b646970
[17179765.492000]        0809c720 c47c8d40 40040c01 0809c720 c016a5f2 
cadb0624 c47c8d40 40040c01
[17179765.492000]        0809c720 c47c8d40 00000000 c016a654 c015845a 
cffefa80 c47c8d40 c016a90d
[17179765.492000] Call Trace:
[17179765.492000]  <d0833913> altpinfo_ioctl+0x103/0x170 [altpinfo] 
<c016a5f2> do_ioctl+0x52/0x60
[17179765.492000]  <c016a654> vfs_ioctl+0x54/0x2b0  <c015845a> 
do_sys_open+0x9a/0xd0
[17179765.492000]  <c016a90d> sys_ioctl+0x5d/0x90  <c0102d27> 
syscall_call+0x7/0xb
[17179765.492000] Code: 5c 24 28 c7 03 f2 ff ff ff e9 12 be f0 ff ba f2 ff 
ff ff e9 6d c0 f0 ff 8d 0c 8a e9 8f c0 f0 ff
01 c1 eb 03 8d 0c 88 51 50 31 c0 <f3> aa 58 59 e9 ca c0 f0 ff 8d 0c 88 51 
50 31 c0 f3 aa 58 59 e9
[17179765.492000] EIP: [<c02c4845>] iret_exc+0x4fd/0x75d SS:ESP 
0068:c32b3f34
[17179765.492000]  <6>altpinfo: unloaded


The function altpinfo_ioctl looks like this:

static int altpinfo_ioctl(struct inode *inode, struct file *f, unsigned 
int cmd, unsigned long arg)
{
         int ret = 0;

         switch (cmd) {

         case API_GET_FMT:
                 ret = put_fmt((char __user *)arg);
                 break;

         case API_SET_FMT:
                 if (!capable(CAP_SYS_ADMIN)) {
                         return -EPERM;
                 }

                 ret = get_fmt((char __user *)arg);
                 break;

         case API_GET_CURLEN:
                 ret = put_curlen(f, (int __user *)arg);
                 break;

         default:
                 ret = -ENOTTY;
                 break;

         }

         return ret;
}


Helper functions look like this:

static int put_curlen(struct file *f, int __user *arg)
{
         int left;
         struct private_data *p = (struct private_data *) f->private_data;

         if (down_interruptible(&p->sem) != 0) {
                 return -ERESTARTSYS;
         }

         left = p->buffer_size - f->f_pos;

         up(&p->sem);

         return put_user(left, arg);
}

static int put_fmt(char __user *arg)
{
         char *fmt;
         int ret;

         ret = get_format_string(&fmt);

         if (ret != 0) {
                 return ret;
         }

         if (copy_to_user(arg, fmt, strlen(fmt)) != 0) {
                 return -EIO;
         }

         kfree(fmt);

         return 0;
}

/* Better make arg buffer at least 4 bytes alligned in userspace! */
static int get_fmt(char __user *arg)
{
         int ret;
         int len;
         char *buf;

         ret = get_user(len, (int __user *)arg);

         if (ret != 0) {
                 return ret;
         }

         buf = kcalloc(len, sizeof(char), GFP_KERNEL);

         if (copy_from_user(buf, arg + 4, len) != 0) {
                 kfree(buf);
                 return -EIO;
         }

         buf[len] = '\0';

         ret = parse_format(buf);

         if (ret != 0) {
                 kfree(buf);
                 return ret;
         }

         return 0;
}


I wonder what does this oops mean and what that function iret_exc really 
does and why? It looks like some automagical exception handler but I am 
not sure about it.

I can't reproduce this oops for now on my machine. Could anybody tell me 
why that code is wrong and why is this oops in iret_exc? (At the same time 
I am waiting for further info from the tester but I am not sure when and 
how much I will get.)

Any help on tracing this down would be appreciated.


Thanks,

Grzegorz Kulewski


^ permalink raw reply	[flat|nested] only message in thread

only message in thread, other threads:[~2007-02-09  2:45 UTC | newest]

Thread overview: (only message) (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2007-02-09  2:26 Strange oops in iret_exc with my own module Grzegorz Kulewski

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).