Commit b769f494 authored by Dan Rosenberg's avatar Dan Rosenberg Committed by Takashi Iwai

sound/oss: remove offset from load_patch callbacks

Was: [PATCH] sound/oss/midi_synth: prevent underflow, use of
uninitialized value, and signedness issue

The offset passed to midi_synth_load_patch() can be essentially
arbitrary.  If it's greater than the header length, this will result in
a copy_from_user(dst, src, negative_val).  While this will just return
-EFAULT on x86, on other architectures this may cause memory corruption.
Additionally, the length field of the sysex_info structure may not be
initialized prior to its use.  Finally, a signed comparison may result
in an unintentionally large loop.

On suggestion by Takashi Iwai, version two removes the offset argument
from the load_patch callbacks entirely, which also resolves similar
issues in opl3.  Compile tested only.

v3 adjusts comments and hopefully gets copy offsets right.
Signed-off-by: default avatarDan Rosenberg <drosenberg@vsecurity.com>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent ce24f58a
...@@ -271,7 +271,7 @@ struct synth_operations ...@@ -271,7 +271,7 @@ struct synth_operations
void (*reset) (int dev); void (*reset) (int dev);
void (*hw_control) (int dev, unsigned char *event); void (*hw_control) (int dev, unsigned char *event);
int (*load_patch) (int dev, int format, const char __user *addr, int (*load_patch) (int dev, int format, const char __user *addr,
int offs, int count, int pmgr_flag); int count, int pmgr_flag);
void (*aftertouch) (int dev, int voice, int pressure); void (*aftertouch) (int dev, int voice, int pressure);
void (*controller) (int dev, int voice, int ctrl_num, int value); void (*controller) (int dev, int voice, int ctrl_num, int value);
void (*panning) (int dev, int voice, int value); void (*panning) (int dev, int voice, int value);
......
...@@ -476,7 +476,7 @@ EXPORT_SYMBOL(midi_synth_hw_control); ...@@ -476,7 +476,7 @@ EXPORT_SYMBOL(midi_synth_hw_control);
int int
midi_synth_load_patch(int dev, int format, const char __user *addr, midi_synth_load_patch(int dev, int format, const char __user *addr,
int offs, int count, int pmgr_flag) int count, int pmgr_flag)
{ {
int orig_dev = synth_devs[dev]->midi_dev; int orig_dev = synth_devs[dev]->midi_dev;
...@@ -491,31 +491,27 @@ midi_synth_load_patch(int dev, int format, const char __user *addr, ...@@ -491,31 +491,27 @@ midi_synth_load_patch(int dev, int format, const char __user *addr,
if (!prefix_cmd(orig_dev, 0xf0)) if (!prefix_cmd(orig_dev, 0xf0))
return 0; return 0;
/* Invalid patch format */
if (format != SYSEX_PATCH) if (format != SYSEX_PATCH)
{
/* printk("MIDI Error: Invalid patch format (key) 0x%x\n", format);*/
return -EINVAL; return -EINVAL;
}
/* Patch header too short */
if (count < hdr_size) if (count < hdr_size)
{
/* printk("MIDI Error: Patch header too short\n");*/
return -EINVAL; return -EINVAL;
}
count -= hdr_size; count -= hdr_size;
/* /*
* Copy the header from user space but ignore the first bytes which have * Copy the header from user space
* been transferred already.
*/ */
if(copy_from_user(&((char *) &sysex)[offs], &(addr)[offs], hdr_size - offs)) if (copy_from_user(&sysex, addr, hdr_size))
return -EFAULT; return -EFAULT;
if (count < sysex.len) /* Sysex record too short */
{ if ((unsigned)count < (unsigned)sysex.len)
/* printk(KERN_WARNING "MIDI Warning: Sysex record too short (%d<%d)\n", count, (int) sysex.len);*/
sysex.len = count; sysex.len = count;
}
left = sysex.len; left = sysex.len;
src_offs = 0; src_offs = 0;
......
...@@ -8,7 +8,7 @@ int midi_synth_open (int dev, int mode); ...@@ -8,7 +8,7 @@ int midi_synth_open (int dev, int mode);
void midi_synth_close (int dev); void midi_synth_close (int dev);
void midi_synth_hw_control (int dev, unsigned char *event); void midi_synth_hw_control (int dev, unsigned char *event);
int midi_synth_load_patch (int dev, int format, const char __user * addr, int midi_synth_load_patch (int dev, int format, const char __user * addr,
int offs, int count, int pmgr_flag); int count, int pmgr_flag);
void midi_synth_panning (int dev, int channel, int pressure); void midi_synth_panning (int dev, int channel, int pressure);
void midi_synth_aftertouch (int dev, int channel, int pressure); void midi_synth_aftertouch (int dev, int channel, int pressure);
void midi_synth_controller (int dev, int channel, int ctrl_num, int value); void midi_synth_controller (int dev, int channel, int ctrl_num, int value);
......
...@@ -820,7 +820,7 @@ static void opl3_hw_control(int dev, unsigned char *event) ...@@ -820,7 +820,7 @@ static void opl3_hw_control(int dev, unsigned char *event)
} }
static int opl3_load_patch(int dev, int format, const char __user *addr, static int opl3_load_patch(int dev, int format, const char __user *addr,
int offs, int count, int pmgr_flag) int count, int pmgr_flag)
{ {
struct sbi_instrument ins; struct sbi_instrument ins;
...@@ -830,11 +830,7 @@ static int opl3_load_patch(int dev, int format, const char __user *addr, ...@@ -830,11 +830,7 @@ static int opl3_load_patch(int dev, int format, const char __user *addr,
return -EINVAL; return -EINVAL;
} }
/* if (copy_from_user(&ins, addr, sizeof(ins)))
* What the fuck is going on here? We leave junk in the beginning
* of ins and then check the field pretty close to that beginning?
*/
if(copy_from_user(&((char *) &ins)[offs], addr + offs, sizeof(ins) - offs))
return -EFAULT; return -EFAULT;
if (ins.channel < 0 || ins.channel >= SBFM_MAXINSTR) if (ins.channel < 0 || ins.channel >= SBFM_MAXINSTR)
......
...@@ -241,7 +241,7 @@ int sequencer_write(int dev, struct file *file, const char __user *buf, int coun ...@@ -241,7 +241,7 @@ int sequencer_write(int dev, struct file *file, const char __user *buf, int coun
return -ENXIO; return -ENXIO;
fmt = (*(short *) &event_rec[0]) & 0xffff; fmt = (*(short *) &event_rec[0]) & 0xffff;
err = synth_devs[dev]->load_patch(dev, fmt, buf, p + 4, c, 0); err = synth_devs[dev]->load_patch(dev, fmt, buf + p, c, 0);
if (err < 0) if (err < 0)
return err; return err;
......
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