Fix: leak in error handling of userspace param parsing
authorFrancis Deslauriers <francis.deslauriers@efficios.com>
Tue, 28 Aug 2018 19:10:12 +0000 (15:10 -0400)
committerJérémie Galarneau <jeremie.galarneau@efficios.com>
Wed, 29 Aug 2018 20:36:06 +0000 (16:36 -0400)
Reported-by: Coverity Scan (1395217 Resource leak)
Signed-off-by: Francis Deslauriers <francis.deslauriers@efficios.com>
Signed-off-by: Jérémie Galarneau <jeremie.galarneau@efficios.com>
src/bin/lttng/commands/enable_events.c

index 48fa614bdc10750a7a2d9ae3639fdbfa741bbf6f..23dff2038f3ef17b1090ba63e186771f9ee5d60a 100644 (file)
@@ -313,6 +313,13 @@ static int parse_userspace_probe_opts(struct lttng_event *ev, char *opt)
                goto end;
        }
 
+       switch (ev->type) {
+       case LTTNG_EVENT_USERSPACE_PROBE:
+               break;
+       default:
+               assert(0);
+       }
+
        /*
         * userspace probe fields are separated by ':'.
         */
@@ -383,37 +390,38 @@ static int parse_userspace_probe_opts(struct lttng_event *ev, char *opt)
        unescaped_target_path = strutils_unescape_string(target_path, 0);
        if (!unescaped_target_path) {
                ret = -LTTNG_ERR_INVALID;
-               goto end_string;
+               goto end_destroy_lookup_method;
        }
 
        /*
         * If there is not forward slash in the path. Walk the $PATH else
         * expand.
         */
-       if (strchr(target_path, '/') == NULL) {
+       if (strchr(unescaped_target_path, '/') == NULL) {
                /* Walk the $PATH variable to find the targeted binary. */
                real_target_path = zmalloc(LTTNG_PATH_MAX * sizeof(char));
                if (!real_target_path) {
                        PERROR("Error allocating path buffer");
                        ret = CMD_ERROR;
-                       goto end_unescaped_string;
+                       goto end_destroy_lookup_method;
                }
-               ret = walk_command_search_path(target_path, real_target_path);
+               ret = walk_command_search_path(unescaped_target_path, real_target_path);
                if (ret) {
                        ERR("Binary not found.");
                        ret = CMD_ERROR;
-                       goto end_free_path;
+                       goto end_destroy_lookup_method;
                }
        } else {
                /*
                 * Expand references to `/./` and `/../`. This function does not check
-                * if the file exists.
+                * if the file exists. This call returns an allocated buffer on
+                * success.
                 */
-               real_target_path = utils_expand_path_keep_symlink(target_path);
+               real_target_path = utils_expand_path_keep_symlink(unescaped_target_path);
                if (!real_target_path) {
                        ERR("Error expanding the path to binary.");
                        ret = CMD_ERROR;
-                       goto end_free_path;
+                       goto end_destroy_lookup_method;
                }
 
                /*
@@ -423,7 +431,7 @@ static int parse_userspace_probe_opts(struct lttng_event *ev, char *opt)
                ret = access(real_target_path, F_OK);
                if (ret) {
                        ret = CMD_ERROR;
-                       goto end_free_path;
+                       goto end_destroy_lookup_method;
                }
        }
 
@@ -468,32 +476,24 @@ static int parse_userspace_probe_opts(struct lttng_event *ev, char *opt)
                break;
        default:
                ret = CMD_ERROR;
-               goto end_free_path;
+               goto end_destroy_lookup_method;
        }
 
-       switch (ev->type) {
-       case LTTNG_EVENT_USERSPACE_PROBE:
-               break;
-       default:
-               assert(0);
-       }
-
-       goto end;
+       /* Successful parsing, now clean up everything and return. */
+       goto end_string;
 
 end_destroy_location:
        lttng_userspace_probe_location_destroy(probe_location);
 end_destroy_lookup_method:
        lttng_userspace_probe_location_lookup_method_destroy(lookup_method);
-end_free_path:
+end_string:
+       strutils_free_null_terminated_array_of_strings(tokens);
        /*
-        * Free path that was allocated by the call to realpath() or when walking
-        * the PATH.
+        * Freeing both char * here makes the error handling simplier. free()
+        * performs not action if the pointer is NULL.
         */
        free(real_target_path);
-end_unescaped_string:
        free(unescaped_target_path);
-end_string:
-       strutils_free_null_terminated_array_of_strings(tokens);
 end:
        return ret;
 }
This page took 0.027606 seconds and 4 git commands to generate.