mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2024-12-26 11:19:32 +03:00
Illumos 5518 - Memory leaks in libzfs import implementation
5518 Memory leaks in libzfs import implementation
Reviewed by: Dan Fields <dan.fields@nexenta.com>
Reviewed by: Matthew Ahrens <mahrens@delphix.com>
Reviewed by: Serghei Samsi <sscdvp@gmail.com>
Approved by: Dan McDonald <danmcd@omniti.com>
References:
https://www.illumos.org/issues/5518
https://github.com/illumos/illumos-gate/commit/078266a
Porting notes:
- One hunk of this change was already applied independently in
commit 4def05f
.
Ported-by: Brian Behlendorf <behlendorf1@llnl.gov>
This commit is contained in:
parent
519129ff4e
commit
0fdd8d6482
@ -19,9 +19,9 @@
|
|||||||
* CDDL HEADER END
|
* CDDL HEADER END
|
||||||
*/
|
*/
|
||||||
/*
|
/*
|
||||||
|
* Copyright 2015 Nexenta Systems, Inc. All rights reserved.
|
||||||
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright (c) 2012 by Delphix. All rights reserved.
|
* Copyright (c) 2012 by Delphix. All rights reserved.
|
||||||
* Copyright 2014 Nexenta Systems, Inc. All rights reserved.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -203,8 +203,10 @@ fix_paths(nvlist_t *nv, name_entry_t *names)
|
|||||||
if ((devid = get_devid(best->ne_name)) == NULL) {
|
if ((devid = get_devid(best->ne_name)) == NULL) {
|
||||||
(void) nvlist_remove_all(nv, ZPOOL_CONFIG_DEVID);
|
(void) nvlist_remove_all(nv, ZPOOL_CONFIG_DEVID);
|
||||||
} else {
|
} else {
|
||||||
if (nvlist_add_string(nv, ZPOOL_CONFIG_DEVID, devid) != 0)
|
if (nvlist_add_string(nv, ZPOOL_CONFIG_DEVID, devid) != 0) {
|
||||||
|
devid_str_free(devid);
|
||||||
return (-1);
|
return (-1);
|
||||||
|
}
|
||||||
devid_str_free(devid);
|
devid_str_free(devid);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -675,8 +677,10 @@ get_configs(libzfs_handle_t *hdl, pool_list_t *pl, boolean_t active_ok)
|
|||||||
nvlist_add_uint64(holey,
|
nvlist_add_uint64(holey,
|
||||||
ZPOOL_CONFIG_ID, c) != 0 ||
|
ZPOOL_CONFIG_ID, c) != 0 ||
|
||||||
nvlist_add_uint64(holey,
|
nvlist_add_uint64(holey,
|
||||||
ZPOOL_CONFIG_GUID, 0ULL) != 0)
|
ZPOOL_CONFIG_GUID, 0ULL) != 0) {
|
||||||
|
nvlist_free(holey);
|
||||||
goto nomem;
|
goto nomem;
|
||||||
|
}
|
||||||
child[c] = holey;
|
child[c] = holey;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1295,7 +1299,6 @@ static nvlist_t *
|
|||||||
zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
||||||
{
|
{
|
||||||
int i, dirs = iarg->paths;
|
int i, dirs = iarg->paths;
|
||||||
DIR *dirp = NULL;
|
|
||||||
struct dirent64 *dp;
|
struct dirent64 *dp;
|
||||||
char path[MAXPATHLEN];
|
char path[MAXPATHLEN];
|
||||||
char *end, **dir = iarg->path;
|
char *end, **dir = iarg->path;
|
||||||
@ -1336,6 +1339,8 @@ zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
|||||||
taskq_t *t;
|
taskq_t *t;
|
||||||
char *rdsk;
|
char *rdsk;
|
||||||
int dfd;
|
int dfd;
|
||||||
|
boolean_t config_failed = B_FALSE;
|
||||||
|
DIR *dirp;
|
||||||
|
|
||||||
/* use realpath to normalize the path */
|
/* use realpath to normalize the path */
|
||||||
if (realpath(dir[i], path) == 0) {
|
if (realpath(dir[i], path) == 0) {
|
||||||
@ -1366,6 +1371,8 @@ zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
|||||||
|
|
||||||
if ((dfd = open64(rdsk, O_RDONLY)) < 0 ||
|
if ((dfd = open64(rdsk, O_RDONLY)) < 0 ||
|
||||||
(dirp = fdopendir(dfd)) == NULL) {
|
(dirp = fdopendir(dfd)) == NULL) {
|
||||||
|
if (dfd >= 0)
|
||||||
|
(void) close(dfd);
|
||||||
zfs_error_aux(hdl, strerror(errno));
|
zfs_error_aux(hdl, strerror(errno));
|
||||||
(void) zfs_error_fmt(hdl, EZFS_BADPATH,
|
(void) zfs_error_fmt(hdl, EZFS_BADPATH,
|
||||||
dgettext(TEXT_DOMAIN, "cannot open '%s'"),
|
dgettext(TEXT_DOMAIN, "cannot open '%s'"),
|
||||||
@ -1417,7 +1424,7 @@ zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
|||||||
cookie = NULL;
|
cookie = NULL;
|
||||||
while ((slice = avl_destroy_nodes(&slice_cache,
|
while ((slice = avl_destroy_nodes(&slice_cache,
|
||||||
&cookie)) != NULL) {
|
&cookie)) != NULL) {
|
||||||
if (slice->rn_config != NULL) {
|
if (slice->rn_config != NULL && !config_failed) {
|
||||||
nvlist_t *config = slice->rn_config;
|
nvlist_t *config = slice->rn_config;
|
||||||
boolean_t matched = B_TRUE;
|
boolean_t matched = B_TRUE;
|
||||||
|
|
||||||
@ -1438,14 +1445,16 @@ zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
|||||||
}
|
}
|
||||||
if (!matched) {
|
if (!matched) {
|
||||||
nvlist_free(config);
|
nvlist_free(config);
|
||||||
config = NULL;
|
} else {
|
||||||
continue;
|
/*
|
||||||
|
* use the non-raw path for the config
|
||||||
|
*/
|
||||||
|
(void) strlcpy(end, slice->rn_name,
|
||||||
|
pathleft);
|
||||||
|
if (add_config(hdl, &pools, path, i+1,
|
||||||
|
slice->rn_num_labels, config) != 0)
|
||||||
|
config_failed = B_TRUE;
|
||||||
}
|
}
|
||||||
/* use the non-raw path for the config */
|
|
||||||
(void) strlcpy(end, slice->rn_name, pathleft);
|
|
||||||
if (add_config(hdl, &pools, path, i+1,
|
|
||||||
slice->rn_num_labels, config))
|
|
||||||
goto error;
|
|
||||||
}
|
}
|
||||||
free(slice->rn_name);
|
free(slice->rn_name);
|
||||||
free(slice);
|
free(slice);
|
||||||
@ -1453,7 +1462,9 @@ zpool_find_import_impl(libzfs_handle_t *hdl, importargs_t *iarg)
|
|||||||
avl_destroy(&slice_cache);
|
avl_destroy(&slice_cache);
|
||||||
|
|
||||||
(void) closedir(dirp);
|
(void) closedir(dirp);
|
||||||
dirp = NULL;
|
|
||||||
|
if (config_failed)
|
||||||
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef HAVE_LIBBLKID
|
#ifdef HAVE_LIBBLKID
|
||||||
@ -1479,14 +1490,10 @@ error:
|
|||||||
|
|
||||||
for (ne = pools.names; ne != NULL; ne = nenext) {
|
for (ne = pools.names; ne != NULL; ne = nenext) {
|
||||||
nenext = ne->ne_next;
|
nenext = ne->ne_next;
|
||||||
if (ne->ne_name)
|
free(ne->ne_name);
|
||||||
free(ne->ne_name);
|
|
||||||
free(ne);
|
free(ne);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dirp)
|
|
||||||
(void) closedir(dirp);
|
|
||||||
|
|
||||||
return (ret);
|
return (ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1844,9 +1851,9 @@ zpool_in_use(libzfs_handle_t *hdl, int fd, pool_state_t *state, char **namestr,
|
|||||||
cb.cb_type = ZPOOL_CONFIG_SPARES;
|
cb.cb_type = ZPOOL_CONFIG_SPARES;
|
||||||
if (zpool_iter(hdl, find_aux, &cb) == 1) {
|
if (zpool_iter(hdl, find_aux, &cb) == 1) {
|
||||||
name = (char *)zpool_get_name(cb.cb_zhp);
|
name = (char *)zpool_get_name(cb.cb_zhp);
|
||||||
ret = TRUE;
|
ret = B_TRUE;
|
||||||
} else {
|
} else {
|
||||||
ret = FALSE;
|
ret = B_FALSE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1860,9 +1867,9 @@ zpool_in_use(libzfs_handle_t *hdl, int fd, pool_state_t *state, char **namestr,
|
|||||||
cb.cb_type = ZPOOL_CONFIG_L2CACHE;
|
cb.cb_type = ZPOOL_CONFIG_L2CACHE;
|
||||||
if (zpool_iter(hdl, find_aux, &cb) == 1) {
|
if (zpool_iter(hdl, find_aux, &cb) == 1) {
|
||||||
name = (char *)zpool_get_name(cb.cb_zhp);
|
name = (char *)zpool_get_name(cb.cb_zhp);
|
||||||
ret = TRUE;
|
ret = B_TRUE;
|
||||||
} else {
|
} else {
|
||||||
ret = FALSE;
|
ret = B_FALSE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -20,8 +20,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
* Copyright 2015 Nexenta Systems, Inc. All rights reserved.
|
||||||
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright 2011 Nexenta Systems, Inc. All rights reserved.
|
|
||||||
* Copyright (c) 2011, 2014 by Delphix. All rights reserved.
|
* Copyright (c) 2011, 2014 by Delphix. All rights reserved.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@ -1783,7 +1783,7 @@ zpool_import_props(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
|
|||||||
thename = origname;
|
thename = origname;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (props) {
|
if (props != NULL) {
|
||||||
uint64_t version;
|
uint64_t version;
|
||||||
prop_flags_t flags = { .create = B_FALSE, .import = B_TRUE };
|
prop_flags_t flags = { .create = B_FALSE, .import = B_TRUE };
|
||||||
|
|
||||||
@ -1791,12 +1791,13 @@ zpool_import_props(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
|
|||||||
&version) == 0);
|
&version) == 0);
|
||||||
|
|
||||||
if ((props = zpool_valid_proplist(hdl, origname,
|
if ((props = zpool_valid_proplist(hdl, origname,
|
||||||
props, version, flags, errbuf)) == NULL) {
|
props, version, flags, errbuf)) == NULL)
|
||||||
return (-1);
|
return (-1);
|
||||||
} else if (zcmd_write_src_nvlist(hdl, &zc, props) != 0) {
|
if (zcmd_write_src_nvlist(hdl, &zc, props) != 0) {
|
||||||
nvlist_free(props);
|
nvlist_free(props);
|
||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
|
nvlist_free(props);
|
||||||
}
|
}
|
||||||
|
|
||||||
(void) strlcpy(zc.zc_name, thename, sizeof (zc.zc_name));
|
(void) strlcpy(zc.zc_name, thename, sizeof (zc.zc_name));
|
||||||
@ -1805,11 +1806,11 @@ zpool_import_props(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
|
|||||||
&zc.zc_guid) == 0);
|
&zc.zc_guid) == 0);
|
||||||
|
|
||||||
if (zcmd_write_conf_nvlist(hdl, &zc, config) != 0) {
|
if (zcmd_write_conf_nvlist(hdl, &zc, config) != 0) {
|
||||||
nvlist_free(props);
|
zcmd_free_nvlists(&zc);
|
||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
if (zcmd_alloc_dst_nvlist(hdl, &zc, zc.zc_nvlist_conf_size * 2) != 0) {
|
if (zcmd_alloc_dst_nvlist(hdl, &zc, zc.zc_nvlist_conf_size * 2) != 0) {
|
||||||
nvlist_free(props);
|
zcmd_free_nvlists(&zc);
|
||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1825,6 +1826,9 @@ zpool_import_props(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
|
|||||||
error = errno;
|
error = errno;
|
||||||
|
|
||||||
(void) zcmd_read_dst_nvlist(hdl, &zc, &nv);
|
(void) zcmd_read_dst_nvlist(hdl, &zc, &nv);
|
||||||
|
|
||||||
|
zcmd_free_nvlists(&zc);
|
||||||
|
|
||||||
zpool_get_rewind_policy(config, &policy);
|
zpool_get_rewind_policy(config, &policy);
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
@ -1936,9 +1940,6 @@ zpool_import_props(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
|
|||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
zcmd_free_nvlists(&zc);
|
|
||||||
nvlist_free(props);
|
|
||||||
|
|
||||||
return (ret);
|
return (ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3335,8 +3336,10 @@ devid_to_path(char *devid_str)
|
|||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
return (NULL);
|
return (NULL);
|
||||||
|
|
||||||
if ((path = strdup(list[0].devname)) == NULL)
|
/*
|
||||||
return (NULL);
|
* In a case the strdup() fails, we will just return NULL below.
|
||||||
|
*/
|
||||||
|
path = strdup(list[0].devname);
|
||||||
|
|
||||||
devid_free_nmlist(list);
|
devid_free_nmlist(list);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user