1
0
mirror of https://git.FreeBSD.org/src.git synced 2024-12-16 10:20:30 +00:00

Fix a null pointer de-ref when passthrough ioctls are issued. This

involves some code shuffle to avoid locking problems.

MFC After: 3 days
This commit is contained in:
Scott Long 2005-03-13 06:25:53 +00:00
parent 17de4e19d8
commit e8447f7027
Notes: svn2git 2020-12-20 02:59:44 +00:00
svn path=/head/; revision=143488

View File

@ -471,11 +471,7 @@ amr_ioctl(struct cdev *dev, u_long cmd, caddr_t addr, int32_t flag, d_thread_t *
/* handle inbound data buffer */ /* handle inbound data buffer */
if (au_length != 0 && au_cmd[0] != 0x06) { if (au_length != 0 && au_cmd[0] != 0x06) {
if ((dp = malloc(au_length, M_DEVBUF, M_WAITOK)) == NULL) dp = malloc(au_length, M_DEVBUF, M_WAITOK|M_ZERO);
return(ENOMEM);
if ((ap = malloc(sizeof(struct amr_passthrough ), M_DEVBUF, M_WAITOK)) == NULL)
return(ENOMEM);
if ((error = copyin(au_buffer, dp, au_length)) != 0) { if ((error = copyin(au_buffer, dp, au_length)) != 0) {
free(dp, M_DEVBUF); free(dp, M_DEVBUF);
@ -484,6 +480,10 @@ amr_ioctl(struct cdev *dev, u_long cmd, caddr_t addr, int32_t flag, d_thread_t *
debug(2, "copyin %ld bytes from %p -> %p", au_length, au_buffer, dp); debug(2, "copyin %ld bytes from %p -> %p", au_length, au_buffer, dp);
} }
/* Allocate this now before the mutex gets held */
if (au_cmd[0] == AMR_CMD_PASS)
ap = malloc(sizeof(struct amr_passthrough), M_DEVBUF, M_WAITOK|M_ZERO);
mtx_lock(&sc->amr_io_lock); mtx_lock(&sc->amr_io_lock);
if ((ac = amr_alloccmd(sc)) == NULL) { if ((ac = amr_alloccmd(sc)) == NULL) {
error = ENOMEM; error = ENOMEM;
@ -563,10 +563,13 @@ amr_ioctl(struct cdev *dev, u_long cmd, caddr_t addr, int32_t flag, d_thread_t *
* At this point, we know that there is a lock held and that these * At this point, we know that there is a lock held and that these
* objects have been allocated. * objects have been allocated.
*/ */
free(dp, M_DEVBUF); if (ac != NULL)
free(ap, M_DEVBUF); amr_releasecmd(ac);
amr_releasecmd(ac);
mtx_unlock(&sc->amr_io_lock); mtx_unlock(&sc->amr_io_lock);
if (dp != NULL)
free(dp, M_DEVBUF);
if (ap != NULL)
free(ap, M_DEVBUF);
return(error); return(error);
} }