* [PATCH 2/2] Eliminate double kfree @ 2008-05-29 13:05 Julia Lawall 2008-05-29 15:33 ` Dmitry Torokhov 0 siblings, 1 reply; 4+ messages in thread From: Julia Lawall @ 2008-05-29 13:05 UTC (permalink / raw) To: dmitry.torokhov, linux-input, linux-kernel, kernel-janitors From: Julia Lawall <julia@diku.dk> The variable report is only non-NULL and non-freed in a small region of code, so it should only be freed in error handling code that comes from that region. This was found using the following semantic match. (http://www.emn.fr/x-info/coccinelle/) // <smpl> @r1@ expression E; position p1,p2; @@ kfree@p1(E); ... kfree@p2(E); @subexps@ expression E1; position r1.p1,p; @@ kfree@p1(<+... E1@p ...+>); @recollect@ position subexps.p; expression E1; @@ E1@p @doublekfree@ position r1.p1,r1.p2; expression recollect.E1,E2,E; position p; statement S; @@ kfree@p1(E); <+... E1@p=E2 ...+> // the actual semantic match contains other assignments kfree@p2(E); @notdoublekfree@ position r1.p1,r1.p2; position any doublekfree.p; expression E,E1,E2; @@ * kfree@p1(E); ... when != E1@p when != E1@p = E2 // needed to match a variable decl * kfree@p2(E); // </smpl> Signed-off-by: Julia Lawall <julia@diku.dk> --- diff -u -p a/drivers/input/tablet/gtco.c b/drivers/input/tablet/gtco.c --- a/drivers/input/tablet/gtco.c 2008-05-09 16:46:57.000000000 +0200 +++ b/drivers/input/tablet/gtco.c 2008-05-29 14:12:31.000000000 +0200 @@ -926,7 +926,7 @@ static int gtco_probe(struct usb_interfa err("Failed to get HID Report Descriptor of size: %d", hid_desc->wDescriptorLength); error = -EIO; - goto err_free_urb; + goto err_free_report; } /* Now we parse the report */ @@ -982,13 +982,14 @@ static int gtco_probe(struct usb_interfa return 0; + err_free_report: + kfree(report); err_free_urb: usb_free_urb(gtco->urbinfo); err_free_buf: usb_buffer_free(gtco->usbdev, REPORT_MAX_SIZE, gtco->buffer, gtco->buf_dma); err_free_devs: - kfree(report); input_free_device(input_dev); kfree(gtco); return error; ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH 2/2] Eliminate double kfree 2008-05-29 13:05 [PATCH 2/2] Eliminate double kfree Julia Lawall @ 2008-05-29 15:33 ` Dmitry Torokhov 2008-05-29 15:57 ` Julia Lawall 0 siblings, 1 reply; 4+ messages in thread From: Dmitry Torokhov @ 2008-05-29 15:33 UTC (permalink / raw) To: Julia Lawall; +Cc: linux-input, linux-kernel, kernel-janitors Hi Julia, On Thu, May 29, 2008 at 03:05:07PM +0200, Julia Lawall wrote: > From: Julia Lawall <julia@diku.dk> > > The variable report is only non-NULL and non-freed in a small region of > code, so it should only be freed in error handling code that comes from > that region. > Thank you for your patch. There isn't a double-kfree though, as far as I can see. Because we need to free the report in both success and failure cases the error handling is a bit unwieldy I agree. I don't want to apply your patch though because I don't like when we bypass parts of error handling path that weren't bypassed if we aborted earlier, if you follow me. What do you think about the patch below? -- Dmitry Input: gtco - clean up error handling in gtco_probe Thanks to Julia Lawall for noticing ugliness. Signed-off-by: Dmitry Torokhov <dtor@mail.ru> --- drivers/input/tablet/gtco.c | 17 +++++++---------- 1 file changed, 7 insertions(+), 10 deletions(-) Index: linux/drivers/input/tablet/gtco.c =================================================================== --- linux.orig/drivers/input/tablet/gtco.c +++ linux/drivers/input/tablet/gtco.c @@ -830,7 +830,7 @@ static int gtco_probe(struct usb_interfa struct gtco *gtco; struct input_dev *input_dev; struct hid_descriptor *hid_desc; - char *report = NULL; + char *report; int result = 0, retry; int error; struct usb_endpoint_descriptor *endpoint; @@ -916,12 +916,16 @@ static int gtco_probe(struct usb_interfa le16_to_cpu(hid_desc->wDescriptorLength), 5000); /* 5 secs */ - if (result == le16_to_cpu(hid_desc->wDescriptorLength)) + dbg("usb_control_msg result: %d", result); + if (result == le16_to_cpu(hid_desc->wDescriptorLength)) { + parse_hid_report_descriptor(gtco, report, result); break; + } } + kfree(report); + /* If we didn't get the report, fail */ - dbg("usb_control_msg result: :%d", result); if (result != le16_to_cpu(hid_desc->wDescriptorLength)) { err("Failed to get HID Report Descriptor of size: %d", hid_desc->wDescriptorLength); @@ -929,12 +933,6 @@ static int gtco_probe(struct usb_interfa goto err_free_urb; } - /* Now we parse the report */ - parse_hid_report_descriptor(gtco, report, result); - - /* Now we delete it */ - kfree(report); - /* Create a device file node */ usb_make_path(gtco->usbdev, gtco->usbpath, sizeof(gtco->usbpath)); strlcat(gtco->usbpath, "/input0", sizeof(gtco->usbpath)); @@ -988,7 +986,6 @@ static int gtco_probe(struct usb_interfa usb_buffer_free(gtco->usbdev, REPORT_MAX_SIZE, gtco->buffer, gtco->buf_dma); err_free_devs: - kfree(report); input_free_device(input_dev); kfree(gtco); return error; ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH 2/2] Eliminate double kfree 2008-05-29 15:33 ` Dmitry Torokhov @ 2008-05-29 15:57 ` Julia Lawall 2008-05-29 16:14 ` Dmitry Torokhov 0 siblings, 1 reply; 4+ messages in thread From: Julia Lawall @ 2008-05-29 15:57 UTC (permalink / raw) To: Dmitry Torokhov; +Cc: linux-input, linux-kernel, kernel-janitors On Thu, 29 May 2008, Dmitry Torokhov wrote: > Hi Julia, > > On Thu, May 29, 2008 at 03:05:07PM +0200, Julia Lawall wrote: > > From: Julia Lawall <julia@diku.dk> > > > > The variable report is only non-NULL and non-freed in a small region of > > code, so it should only be freed in error handling code that comes from > > that region. > > > > Thank you for your patch. There isn't a double-kfree though, as far > as I can see. There was a double free because the error handling at the end of the function (for input_register_device) would fall through to the kfree(report) under err_free_devs, even though report is already freed at that point. Your solution does indeed look more elegant, and seems to preserve the semantics of the original code. julia > Because we need to free the report in both success and > failure cases the error handling is a bit unwieldy I agree. I don't > want to apply your patch though because I don't like when we bypass > parts of error handling path that weren't bypassed if we aborted > earlier, if you follow me. > > What do you think about the patch below? > > -- > Dmitry > > Input: gtco - clean up error handling in gtco_probe > > Thanks to Julia Lawall for noticing ugliness. > > Signed-off-by: Dmitry Torokhov <dtor@mail.ru> > --- > drivers/input/tablet/gtco.c | 17 +++++++---------- > 1 file changed, 7 insertions(+), 10 deletions(-) > > Index: linux/drivers/input/tablet/gtco.c > =================================================================== > --- linux.orig/drivers/input/tablet/gtco.c > +++ linux/drivers/input/tablet/gtco.c > @@ -830,7 +830,7 @@ static int gtco_probe(struct usb_interfa > struct gtco *gtco; > struct input_dev *input_dev; > struct hid_descriptor *hid_desc; > - char *report = NULL; > + char *report; > int result = 0, retry; > int error; > struct usb_endpoint_descriptor *endpoint; > @@ -916,12 +916,16 @@ static int gtco_probe(struct usb_interfa > le16_to_cpu(hid_desc->wDescriptorLength), > 5000); /* 5 secs */ > > - if (result == le16_to_cpu(hid_desc->wDescriptorLength)) > + dbg("usb_control_msg result: %d", result); > + if (result == le16_to_cpu(hid_desc->wDescriptorLength)) { > + parse_hid_report_descriptor(gtco, report, result); > break; > + } > } > > + kfree(report); > + > /* If we didn't get the report, fail */ > - dbg("usb_control_msg result: :%d", result); > if (result != le16_to_cpu(hid_desc->wDescriptorLength)) { > err("Failed to get HID Report Descriptor of size: %d", > hid_desc->wDescriptorLength); > @@ -929,12 +933,6 @@ static int gtco_probe(struct usb_interfa > goto err_free_urb; > } > > - /* Now we parse the report */ > - parse_hid_report_descriptor(gtco, report, result); > - > - /* Now we delete it */ > - kfree(report); > - > /* Create a device file node */ > usb_make_path(gtco->usbdev, gtco->usbpath, sizeof(gtco->usbpath)); > strlcat(gtco->usbpath, "/input0", sizeof(gtco->usbpath)); > @@ -988,7 +986,6 @@ static int gtco_probe(struct usb_interfa > usb_buffer_free(gtco->usbdev, REPORT_MAX_SIZE, > gtco->buffer, gtco->buf_dma); > err_free_devs: > - kfree(report); > input_free_device(input_dev); > kfree(gtco); > return error; > -- > To unsubscribe from this list: send the line "unsubscribe kernel-janitors" in > the body of a message to majordomo@vger.kernel.org > More majordomo info at http://vger.kernel.org/majordomo-info.html > ^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH 2/2] Eliminate double kfree 2008-05-29 15:57 ` Julia Lawall @ 2008-05-29 16:14 ` Dmitry Torokhov 0 siblings, 0 replies; 4+ messages in thread From: Dmitry Torokhov @ 2008-05-29 16:14 UTC (permalink / raw) To: Julia Lawall; +Cc: linux-input, linux-kernel, kernel-janitors On Thu, May 29, 2008 at 05:57:50PM +0200, Julia Lawall wrote: > On Thu, 29 May 2008, Dmitry Torokhov wrote: > > > Hi Julia, > > > > On Thu, May 29, 2008 at 03:05:07PM +0200, Julia Lawall wrote: > > > From: Julia Lawall <julia@diku.dk> > > > > > > The variable report is only non-NULL and non-freed in a small region of > > > code, so it should only be freed in error handling code that comes from > > > that region. > > > > > > > Thank you for your patch. There isn't a double-kfree though, as far > > as I can see. > > There was a double free because the error handling at the end of the > function (for input_register_device) would fall through to the > kfree(report) under err_free_devs, even though report is already freed at > that point. > Ah, yes, indeed.. Well, that decides what branch the patch gets applied then ;) -- Dmitry ^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2008-05-29 16:14 UTC | newest] Thread overview: 4+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2008-05-29 13:05 [PATCH 2/2] Eliminate double kfree Julia Lawall 2008-05-29 15:33 ` Dmitry Torokhov 2008-05-29 15:57 ` Julia Lawall 2008-05-29 16:14 ` Dmitry Torokhov
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).