9a49c8ebd7
Add the ability to to parse sysfs paths to sysctl nodes by replacing '/' with '.' Submitted by: slavash@ MFC after: 1 week Sponsored by: Mellanox Technologies
267 lines
6.2 KiB
C
267 lines
6.2 KiB
C
/*
|
|
* Copyright (c) 2004-2008 Voltaire Inc. All rights reserved.
|
|
*
|
|
* This software is available to you under a choice of one of two
|
|
* licenses. You may choose to be licensed under the terms of the GNU
|
|
* General Public License (GPL) Version 2, available from the file
|
|
* COPYING in the main directory of this source tree, or the
|
|
* OpenIB.org BSD license below:
|
|
*
|
|
* Redistribution and use in source and binary forms, with or
|
|
* without modification, are permitted provided that the following
|
|
* conditions are met:
|
|
*
|
|
* - Redistributions of source code must retain the above
|
|
* copyright notice, this list of conditions and the following
|
|
* disclaimer.
|
|
*
|
|
* - Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the following
|
|
* disclaimer in the documentation and/or other materials
|
|
* provided with the distribution.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*
|
|
*/
|
|
#include <config.h>
|
|
|
|
#include <infiniband/endian.h>
|
|
#include <inttypes.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/sysctl.h>
|
|
#include <fcntl.h>
|
|
#include <dirent.h>
|
|
#include "sysfs.h"
|
|
|
|
static int ret_code(void)
|
|
{
|
|
int e = errno;
|
|
|
|
if (e > 0)
|
|
return -e;
|
|
return e;
|
|
}
|
|
|
|
int sys_read_string(const char *dir_name, const char *file_name, char *str, int max_len)
|
|
{
|
|
char path[256], *s;
|
|
size_t len;
|
|
|
|
snprintf(path, sizeof(path), "%s/%s", dir_name, file_name);
|
|
|
|
len = max_len;
|
|
if (sysctlbyname(PATH_TO_SYS(path), str, &len, NULL, 0) == -1)
|
|
return ret_code();
|
|
|
|
str[(len < max_len) ? len : max_len - 1] = 0;
|
|
|
|
if ((s = strrchr(str, '\n')))
|
|
*s = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sys_read_guid(const char *dir_name, const char *file_name, __be64 *net_guid)
|
|
{
|
|
char buf[32], *str, *s;
|
|
uint64_t guid;
|
|
int r, i;
|
|
|
|
if ((r = sys_read_string(dir_name, file_name, buf, sizeof(buf))) < 0)
|
|
return r;
|
|
|
|
guid = 0;
|
|
|
|
for (s = buf, i = 0; i < 4; i++) {
|
|
if (!(str = strsep(&s, ": \t\n")))
|
|
return -EINVAL;
|
|
guid = (guid << 16) | (strtoul(str, NULL, 16) & 0xffff);
|
|
}
|
|
|
|
*net_guid = htobe64(guid);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sys_read_gid(const char *dir_name, const char *file_name,
|
|
union umad_gid *gid)
|
|
{
|
|
char buf[64], *str, *s;
|
|
__be16 *ugid = (__be16 *) gid;
|
|
int r, i;
|
|
|
|
if ((r = sys_read_string(dir_name, file_name, buf, sizeof(buf))) < 0)
|
|
return r;
|
|
|
|
for (s = buf, i = 0; i < 8; i++) {
|
|
if (!(str = strsep(&s, ": \t\n")))
|
|
return -EINVAL;
|
|
ugid[i] = htobe16(strtoul(str, NULL, 16) & 0xffff);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sys_read_uint64(const char *dir_name, const char *file_name, uint64_t * u)
|
|
{
|
|
char buf[32];
|
|
int r;
|
|
|
|
if ((r = sys_read_string(dir_name, file_name, buf, sizeof(buf))) < 0)
|
|
return r;
|
|
|
|
*u = strtoull(buf, NULL, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sys_read_uint(const char *dir_name, const char *file_name, unsigned *u)
|
|
{
|
|
char buf[32];
|
|
int r;
|
|
|
|
if ((r = sys_read_string(dir_name, file_name, buf, sizeof(buf))) < 0)
|
|
return r;
|
|
|
|
*u = strtoul(buf, NULL, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define DIRECTSIZ(namlen) \
|
|
(((uintptr_t)&((struct dirent *)0)->d_name + \
|
|
((namlen)+1)*sizeof(((struct dirent *)0)->d_name[0]) + 3) & ~3)
|
|
|
|
int
|
|
sys_scandir(const char *dirname, struct dirent ***namelist,
|
|
int (*select)(const struct dirent *),
|
|
int (*compar)(const struct dirent **, const struct dirent **))
|
|
{
|
|
struct dirent **names;
|
|
struct dirent **names2;
|
|
struct dirent *dp;
|
|
char name[1024];
|
|
int lsname[22];
|
|
int chname[22];
|
|
int name2[22];
|
|
int oid[22];
|
|
char *s;
|
|
size_t n1, n2;
|
|
size_t len, oidlen, namlen;
|
|
int cnt, max;
|
|
int err;
|
|
int i;
|
|
|
|
*namelist = NULL;
|
|
if (strlcpy(name, PATH_TO_SYS(dirname), sizeof(name)) >= sizeof(name))
|
|
return (-EINVAL);
|
|
/*
|
|
* Resolve the path.
|
|
*/
|
|
len = sizeof(oid) / sizeof(int);
|
|
namlen = strlen(name) + 1;
|
|
if (sysctlnametomib(name, oid, &len) != 0)
|
|
return (-errno);
|
|
lsname[0] = 0; /* Root */
|
|
lsname[1] = 2; /* Get next */
|
|
memcpy(lsname+2, oid, len * sizeof(int));
|
|
n1 = 2 + len;
|
|
oidlen = len;
|
|
/*
|
|
* Setup the return list of dirents.
|
|
*/
|
|
cnt = 0;
|
|
max = 64;
|
|
names = malloc(max * sizeof(void *));
|
|
if (names == NULL)
|
|
return (-ENOMEM);
|
|
|
|
for (;;) {
|
|
n2 = sizeof(name2);
|
|
if (sysctl(lsname, n1, name2, &n2, 0, 0) < 0) {
|
|
if (errno == ENOENT)
|
|
break;
|
|
goto errout;
|
|
}
|
|
n2 /= sizeof(int);
|
|
if (n2 < oidlen)
|
|
break;
|
|
for (i = 0; i < oidlen; i++)
|
|
if (name2[i] != oid[i])
|
|
goto out;
|
|
chname[0] = 0; /* root */
|
|
chname[1] = 1; /* oid name */
|
|
memcpy(chname + 2, name2, n2 * sizeof(int));
|
|
memcpy(lsname + 2, name2, n2 * sizeof(int));
|
|
n1 = 2 + n2;
|
|
/*
|
|
* scandir() is not supposed to go deeper than the requested
|
|
* directory but sysctl also doesn't return a node for
|
|
* 'subdirectories' so we have to find a file in the subdir
|
|
* and then truncate the name to report it.
|
|
*/
|
|
if (n2 > oidlen + 1) {
|
|
/* Skip to the next name after this one. */
|
|
n1 = 2 + oidlen + 1;
|
|
lsname[n1 - 1]++;
|
|
}
|
|
len = sizeof(name);
|
|
if (sysctl(chname, n2 + 2, name, &len, 0, 0) < 0)
|
|
goto errout;
|
|
if (len <= 0 || len < namlen)
|
|
goto out;
|
|
s = name + namlen;
|
|
/* Just keep the first level name. */
|
|
if (strchr(s, '.'))
|
|
*strchr(s, '.') = '\0';
|
|
len = strlen(s) + 1;
|
|
dp = malloc(DIRECTSIZ(len));
|
|
dp->d_reclen = DIRECTSIZ(len);
|
|
dp->d_namlen = len;
|
|
memcpy(&dp->d_name, s, len);
|
|
if (select && !select(dp)) {
|
|
free(dp);
|
|
continue;
|
|
}
|
|
if (cnt == max) {
|
|
max *= 2;
|
|
names2 = realloc(names, max * sizeof(void *));
|
|
if (names2 == NULL) {
|
|
errno = ENOMEM;
|
|
free(dp);
|
|
goto errout;
|
|
}
|
|
names = names2;
|
|
}
|
|
names[cnt++] = dp;
|
|
}
|
|
out:
|
|
if (cnt && compar)
|
|
qsort(names, cnt, sizeof(struct dirent *),
|
|
(int (*)(const void *, const void *))compar);
|
|
|
|
*namelist = names;
|
|
|
|
return (cnt);
|
|
|
|
errout:
|
|
err = errno;
|
|
for (i = 0; i < cnt; i++)
|
|
free(names[i]);
|
|
free(names);
|
|
return (-err);
|
|
}
|