Index: lib/libutil/Makefile =================================================================== --- lib/libutil/Makefile +++ lib/libutil/Makefile @@ -15,11 +15,11 @@ hexdump.c humanize_number.c kinfo_getfile.c \ kinfo_getallproc.c kinfo_getproc.c kinfo_getvmmap.c \ kinfo_getvmobject.c kld.c \ - login_auth.c login_cap.c \ + libsysctl.c login_auth.c login_cap.c \ login_class.c login_crypt.c login_ok.c login_times.c login_tty.c \ pidfile.c property.c pty.c pw_scan.c pw_util.c quotafile.c \ realhostname.c stub.c trimdomain.c uucplock.c -INCS= libutil.h login_cap.h +INCS= libutil.h libsysctl.h login_cap.h CFLAGS+= -DNO__SCCSID @@ -32,7 +32,7 @@ MAN+= expand_number.3 flopen.3 fparseln.3 hexdump.3 \ humanize_number.3 kinfo_getallproc.3 kinfo_getfile.3 \ kinfo_getproc.3 kinfo_getvmmap.3 kinfo_getvmobject.3 kld.3 \ - login_auth.3 login_cap.3 \ + libsysctl.3 login_auth.3 login_cap.3 \ login_class.3 login_ok.3 login_times.3 login_tty.3 pidfile.3 \ property.3 pty.3 quotafile.3 realhostname.3 realhostname_sa.3 \ _secure_path.3 trimdomain.3 uucplock.3 pw_util.3 Index: lib/libutil/libsysctl.h =================================================================== --- lib/libutil/libsysctl.h +++ lib/libutil/libsysctl.h @@ -0,0 +1,151 @@ +/*- + * SPDX-License-Identifier: BSD-2-Clause-FreeBSD + * + * Copyright (c) 2018-2019 Alfonso Sabato Siciliano + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. 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. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + + +#ifndef _LIBSYSCTL_H +#define _LIBSYSCTL_H + +#include +#include +#include + +#define LIBSYSCTL_VERSION 1 +#define LIBSYSCTL_IDMAXLEVEL (CTL_MAXNAME - 2) + + +/* + * functions to wrap 'undocumented kern_sysctl.c API', + * return: 0 for success, negative value for failure. + */ + +int +libsysctl_nametoid(const char *name, size_t namelen, int *id, size_t *idlevel); + +int libsysctl_name(int *id, size_t idlevel, char *name, size_t *namelen); +int libsysctl_desc(int *id, size_t idlevel, char *desc, size_t *desclen); +int libsysctl_label(int *id, size_t idlevel, char *label, size_t *labellen); + +#define LIBSYSCTL_NAMELEN(id, idlevel, size) \ + libsysctl_name(id, idlevel, NULL, size) +#define LIBSYSCTL_DESCLEN(id, idlevel, size) \ + libsysctl_desc(id, idlevel, NULL, size) +#define LIBSYSCTL_LABELLEN(id, idlevel, size) \ + libsysctl_label(id, idlevel, NULL, size) + +int libsysctl_info(int *id, size_t idlevel, void *info, size_t *infolen); + +#define LIBSYSCTL_INFOKIND(info) (*((unsigned int *)info)) +#define LIBSYSCTL_INFOTYPE(info) (*((unsigned int *)info) & CTLTYPE) +#define LIBSYSCTL_INFOFLAGS(info) (*((unsigned int *)info) & 0xfffffff0) +#define LIBSYSCTL_INFOFMT(info) ((char *)info + sizeof(unsigned int)) + +/* kernel returns only next leaf, next node requires extra computation */ +int +libsysctl_nextnode(int *id, size_t idlevel, int *idnext, size_t *idnextlevel); +int +libsysctl_nextleaf(int *id, size_t idlevel, int *idnext, size_t *idnextlevel); + + +/* + * functions related to "struct libsysctl_object" + * params: id and idlevel to identify a mib entry + * return: NULL for failure, pointer to allocated memory for success. + */ + +/* 'struct libsysctl_object': userspace mib entry definition */ + +SLIST_HEAD(libsysctl_object_list, libsysctl_object); + +struct libsysctl_object { + SLIST_ENTRY(libsysctl_object) object_link; + int *id; + size_t idlevel; /* between 1 and LIBSYSCTL_IDMAXLEVEL */ + char *name; + char *desc; + char *label; /* aggregation label */ + uint8_t type; /* defined in */ + uint32_t flags; /* defined in */ + char *fmt; /* format string */ + /* children is set by libsysctl_tree() */ + struct libsysctl_object_list *children; +}; + +/* + * OR_FLAGS: object fields to set, + * .id and .idlevel are always set + * .children is default for libsysctl_tree() + */ +#define LIBSYSCTL_FNAME 0x01 /* .name */ +#define LIBSYSCTL_FDESC 0x02 /* .desc */ +#define LIBSYSCTL_FLABEL 0x04 /* .label */ +#define LIBSYSCTL_FTYPE 0x08 /* .type */ +#define LIBSYSCTL_FFLAGS 0x10 /* .flags */ +#define LIBSYSCTL_FFMT 0x20 /* .fmt */ +#define LIBSYSCTL_FALL /* all */ \ + LIBSYSCTL_FNAME | LIBSYSCTL_FDESC \ + | LIBSYSCTL_FLABEL | LIBSYSCTL_FTYPE \ + | LIBSYSCTL_FFLAGS | LIBSYSCTL_FFMT + + +/* object functions */ + +struct libsysctl_object * +libsysctl_object(int *id, size_t idlevel, unsigned int flags); + +void +libsysctl_freeobject(struct libsysctl_object *object); + + +/* list functions */ + +#define LIBSYSCTL_MAXDEPTH (CTL_MAXNAME - 3) + +typedef int libsysctl_filterfunc_t (struct libsysctl_object *object); + +struct libsysctl_object_list * +libsysctl_filterlist(libsysctl_filterfunc_t *filterfunc, unsigned int flags); + +#define LIBSYSCTL_LIST(flags) libsysctl_filterlist(NULL, flags) + +struct libsysctl_object_list * +libsysctl_grouplist(int *id, size_t idlevel, unsigned int flags, + unsigned int max_depth); + +void +libsysctl_freelist(struct libsysctl_object_list *list); + + +/* tree fuctions */ + +struct libsysctl_object * +libsysctl_tree(int *id, size_t idlevel, unsigned int flags, + unsigned int max_depth); + +void +libsysctl_freetree(struct libsysctl_object *object_root); + +#endif /* _LIBSYSCTL_H */ Index: lib/libutil/libsysctl.3 =================================================================== --- lib/libutil/libsysctl.3 +++ lib/libutil/libsysctl.3 @@ -0,0 +1,418 @@ +.\" +.\" Copyright (c) 2018-2019 Alfonso Sabato Siciliano +.\" +.\" Redistribution and use in source and binary forms, with or without +.\" modification, are permitted provided that the following conditions +.\" are met: +.\" 1. Redistributions of source code must retain the above copyright +.\" notice, this list of conditions and the following disclaimer. +.\" 2. 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. +.\" +.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND +.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +.\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE +.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS +.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) +.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY +.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF +.\" SUCH DAMAGE. +.\" +.Dd January 11, 2019 +.Dt LIBSYSCTL 3 +.Os +.Sh NAME +.Nm LIBSYSCTL_IDMAXLEVEL , +.Nm libsysctl_nametoid , +.Nm libsysctl_name , +.Nm LIBSYSCTL_NAMELEN , +.Nm libsysctl_desc , +.Nm LIBSYSCTL_DESCLEN , +.Nm libsysctl_label , +.Nm LIBSYSCTL_LABELLEN , +.Nm libsysctl_info , +.Nm LIBSYSCTL_INFOKIND , +.Nm LIBSYSCTL_INFOTYPE , +.Nm LIBSYSCTL_INFOFLAGS , +.Nm LIBSYSCTL_INFOFMT , +.Nm libsysctl_nextnode , +.Nm libsysctl_nextleaf , +.Nm libsysctl_object , +.Nm libsysctl_freeobject , +.Nm libsysctl_filterlist , +.Nm LIBSYSCTL_LIST , +.Nm LIBSYSCTL_MAXDEPTH , +.Nm libsysctl_grouplist , +.Nm libsysctl_freelist , +.Nm libsysctl_tree , +.Nm libsysctl_freetree +.Nd manage sysctl mib tree in userpace +.Sh LIBRARY +.Lb libutil +.Sh SYNOPSIS +.In sys/types.h +.In sys/queue.h +.In libsysctl.h +.Fd #define LIBSYSCTL_IDMAXLEVEL; +.Ft "int" +.Fo libsysctl_nametoid +.Fa "const char *name" +.Fa "size_t namelen" +.Fa "int *id" +.Fa "size_t *idlevel" +.Fc +.Ft "int" +.Fn libsysctl_name "int *id" "size_t idlevel" "char *name" "size_t *namelen" +.Ft "int" +.Fn LIBSYSCTL_NAMELEN "int *id" "size_t idlevel" "size_t *namelen" +.Ft "int" +.Fn libsysctl_desc "int *id" "size_t idlevel" "char *desc" "size_t *desclen" +.Ft "int" +.Fn LIBSYSCTL_DESCLEN "int *id" "size_t idlevel" "size_t *desclen" +.Ft "int" +.Fn libsysctl_label "int *id" "size_t idlevel" "char *label" "size_t *labellen" +.Ft "int" +.Fn LIBSYSCTL_LABELLEN "int *id" "size_t idlevel" "size_t *labellen" +.Ft "int" +.Fn libsysctl_info "int *id" "size_t idlevel" "void *info" "size_t *infolen" +.Ft "uint32_t" +.Fn LIBSYSCTL_INFOKIND "info" +.Ft "uint8_t" +.Fn LIBSYSCTL_INFOTYPE "info" +.Ft "uint32_t" +.Fn LIBSYSCTL_INFOFLAGS "info" +.Ft "char *" +.Fn LIBSYSCTL_INFOFMT "info" +.Ft "int" +.Fo libsysctl_nextleaf +.Fa "int *id" +.Fa "size_t idlevel" +.Fa "int *nextid" +.Fa "size_t *nextidlevel" +.Fc +.Ft "int" +.Fo libsysctl_nextnode +.Fa "int *id" +.Fa "size_t idlevel" +.Fa "int *nextid" +.Fa "size_t *nextidlevel" +.Fc +.Ft "struct libsysctl_object *" +.Fn libsysctl_object "int *id" "size_t idlevel" "unsigned int flags" +.Ft "void" +.Fn libsysctl_freeobject "struct libsysctl_object *object" +.Fd #define LIBSYSCTL_MAXDEPTH +.Ft "struct libsysctl_object_list *" +.Fn libsysctl_filterlist "libsysctl_filterfunc_t *filterfunc" "unsigned int flags" +.Ft "struct libsysctl_object_list *" +.Fn LIBSYSCTL_LIST "unsigned int flags" +.Ft "struct libsysctl_object_list *" +.Fo libsysctl_grouplist +.Fa "int *idstart" +.Fa "size_t idstartlen" +.Fa "unsigned int flags" +.Fa "unsigned int max_depth" +.Fc +.Ft "void" +.Fn libsysctl_freelist "struct libsysctl_object_list *list" +.Ft "struct libsysctl_object *" +.Fo libsysctl_tree +.Fa "int *id" +.Fa "size_t idlevel" +.Fa "unsigned int flags" +.Fa "unsigned int max_depth" +.Fc +.Ft "void" +.Fn libsysctl_freetree "struct libsysctl_object *node" +.Sh DESCRIPTION +The +.Nm libsysctl +is an interface to the kernel sysctl-mib-tree. +It implements wrappers around undocumented +.Dq sysctl.* +kernel states to get mib info and +provides a convenient API to build a mib-entry, +entries-list and mib-tree in userspace. +.Pp +A mib-entry is identified by a pair +.Fa "int *id" +and +.Fa "size_t idlevel" , +the level should be between 1 and +.Dv LIBSYSCTL_IDMAXLEVEL , +see +.Fn libsysctl_next ; +.Nm libsysctl +is not designed to get and set entry value, use +.Xr sysctl 3 . +.Pp +.Fn libsysctl_nametoid +sets +.Em id +and +.Em idlevel +like the entry with +.Em name +and +.Em namelen . +.Pp +.Fn libsysctl_name , +.Fn libsysctl_desc +and +.Fn libsysctl_label +set +.Em name +and +.Em namelen , +.Em desc +and +.Em desclen , +.Em label +and +.Em labellen +like the entry with +.Em id +and +.Em idlevel . +.Pp +.Fn LIBSYSCTL_NAMELEN , +.Fn LIBSYSCTL_DESCLEN +and +.Fn LIBSYSCTL_LABELLEN +set +.Em namelen , +.Em desclen , +and +.Em labellen +like the entry with +.Em id +and +.Em idlevel . +.Pp +.Fn libsysctl_info +sets +.Em info +and +.Em infolen +like the entry with +.Em id +and +.Em idlevel , +.Em info +has not a basic format: +3 bytes for flags, 1 byte for type and a char* for the +.Dq format string ; +flags and type are defined in +.In sys/sysctl.h . +Macros to deal with +.Em info : +.Bd -ragged -offset indent -compact +.Fn LIBSYSCTL_INFOFLAGS info +returns flags; +.Ed +.Bd -ragged -offset indent -compact +.Fn LIBSYSCTL_INFOTYPE info +returns entry type; +.Ed +.Bd -ragged -offset indent -compact +.Fn LIBSYSCTL_INFOKIND info +returns flags following by type; +.Ed +.Bd -ragged -offset indent -compact +.Fn LIBSYSCTL_INFOFMT info +returns a pointer to the +.Dq format string . +.Ed +.Pp +.Fn libsysctl_nextleaf +sets +.Em nextid +and +.Em nextidlevel +like the next-leaf-entry of the entry with +.Em id +and +.Em idlevel . +.Fn libsysctl_nextnode +sets +.Em nextid +and +.Em nextidlevel +like the next-[node|leaf]-entry of the entry with +.Em id +and +.Em idlevel . +Notes: +.Em nextid +should have size LIBSYSCTL_IDMAXLEVEL and +.Em nextidlevel +should be set to LIBSYSCTL_IDMAXLEVEL before to call +.Fn libsysctl_nextleaf +or +.Fn libsysctl_nextnode . +.Pp +.Fn libsysctl_object +returns a pointer to allocated memory for a +.Em struct libsysctl_object +(setting +.Em flags +members) of the entry with +.Em id +and +.Em idlevel . +A mib userspace entry is defined: +.Pp +.Bd -literal -offset indent -compact +/* 'struct libsysctl_object': userspace mib entry definition */ + +SLIST_HEAD(libsysctl_object_list, libsysctl_object); + +struct libsysctl_object { + SLIST_ENTRY(libsysctl_object) object_link; + int *id; + size_t idlevel; /* between 1 and LIBSYSCTL_IDMAXLEVEL */ + char *name; + char *desc; + char *label; /* aggregation label */ + uint8_t type; /* defined in */ + uint32_t flags; /* defined in */ + char *fmt; /* format string */ + /* children is set by libsysctl_tree() */ + struct libsysctl_object_list *children; +}; + +/* + * OR_FLAGS: object fields to set, + * .id and .idlevel are always set + * .children is default for libsysctl_tree() + */ +#define LIBSYSCTL_FNAME 0x01 /* .name */ +#define LIBSYSCTL_FDESC 0x02 /* .desc */ +#define LIBSYSCTL_FLABEL 0x04 /* .label */ +#define LIBSYSCTL_FTYPE 0x08 /* .type */ +#define LIBSYSCTL_FFLAGS 0x10 /* .flags */ +#define LIBSYSCTL_FFMT 0x20 /* .fmt */ +#define LIBSYSCTL_FALL /* all */ +.Ed +.Pp +.Fn LIBSYSCTL_LIST +allocates memory and returns a SLIST of libsysctl_object (setting +.Em flags +members), it is an alias for +.Fn libsysctl_filterlist NULL flags . +.Pp +.Fn libsysctl_filterlist +allocates memory for a SLIST of libsysctl_object (setting +.Em flags +members), an object +.Dq o +is added if +.Bd -ragged -offset indent -compact +int libsysctl_filterfunc_t (struct libsysctl_object *o) +.Ed +returns 0 or +.Em filterfunc +is NULL; notes: +.Fn libsysctl_filterlist +uses +.Fn libsysctl_nextnode +and object.children is not set. +.Pp +.Fn libsysctl_grouplist +allocates memory and returns a SLIST of libsysctl_object (setting +.Em flags +members) visited in a +.Dq Depth First Traversal +until +.Em max_depth , +.Em id +and +.Em idlevel +denote the root. +Notes: +.Fn libsysctl_grouplist +uses +.Fn libsysctl_nextnode , +object.children is not set and +.Em max_depth +can be set to +.Dv LIBSYSCTL_MAXDEPTH . +.Pp +.Fn libsysctl_tree +allocates memory for a tree of libsysctl_object nodes (setting +.Em flags +members) until +.Em max_depth +and returns a pointer to the root: the entry with +.Em id +and +.Em idlevel . +Notes: +.Em max_depth +can be set to +.Dv LIBSYSCTL_MAXDEPTH , +object.children is set and iterable by SLIST macros. +.Pp +.Fn libsysctl_freeobject , +.Fn libsysctl_freelist +and +.Fn libsysctl_freetree +free allocated memory. +.Sh IMPLEMENTATION NOTES +.Nm libsysctl +calls +.Fn sysctl +syscall for wrapping 0.[1-6] entries defined in kern_sysctl.c. +Kernel returns only next leaf, +.Fn libsysctl_nextnode +requires extra computation. +.Sh RETURN VALUES +.Rv -std libsysctl_nametoid LIBSYSCTL_NAMELEN libsysctl_name LIBSYSCTL_DESCLEN libsysctl_desc LIBSYSCTL_LABELLEN libsysctl_label libsysctl_info libsysctl_nextnode libsysctl_nextleaf +.Pp +The +.Fn libsysctl_object , +.Fn libsysctl_filterlist , +.Fn LIBSYSCTL_LIST , +.Fn libsysctl_grouplist , +.Fn libsysctl_tree +functions return +.Dv NULL +upon error or a pointer to allocated memory for success. +.Sh EXAMPLES +Complete set of examples: +.Dl https://gitlab.com/alfix/libsysctl/tree/master/examples +.\" .Sh ERRORS +.\" For sections 2, 3, 4, and 9 errno settings only. +.Sh SEE ALSO +.Xr queue 3 , +.Xr sysctl 3 +.Sh HISTORY +The +.Nm libsysctl +library first appeared in +.Fx 13.0 . +.Sh AUTHORS +.Nm libsysctl +and this manual page were written by +.An Alfonso S. Siciliano Aq Mt alf.siciliano@gmail.com +.\" .Sh CAVEATS +.Sh BUGS +Problems from kern_sysctl.c: +.Bd -ragged -offset indent -compact +.Fn libsysctl_name +false positive. +.Ed +.Bd -ragged -offset indent -compact +.Fn libsysctl_desc +could set +.Em desc +to: +.Dq +or +.Dv NULL +for entries without description. +.Ed Index: lib/libutil/libsysctl.c =================================================================== --- lib/libutil/libsysctl.c +++ lib/libutil/libsysctl.c @@ -0,0 +1,541 @@ +/*- + * SPDX-License-Identifier: BSD-2-Clause-FreeBSD + * + * Copyright (c) 2018-2019 Alfonso Sabato Siciliano + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. 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. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + + +#include +#include +#include + +#include +#include +#include +#include + + +/* + * sys/sysctl.h lacks these identifiers + */ + +/* Top-level identifiers */ +#define CTL_SYSCTLMIB 0 + +/* CTL_SYSCTLMIB identifiers */ +#define MIB_OBJECTNAME 1 +#define MIB_NEXTOID 2 +#define MIB_NAME2OID 3 +#define MIB_OBJECTFMT 4 +#define MIB_OBJECTDESCR 5 +#define MIB_OBJECTLABEL 6 + + +/* Internal use */ + +static int +libsysctl_internal_subtree(struct libsysctl_object *obj, unsigned int flags, + unsigned int depth) +{ + int error = 0; + struct libsysctl_object *child; + + if ((obj->children = + libsysctl_grouplist(obj->id, obj->idlevel, flags, 1)) == NULL) { + return (1); + } + + SLIST_REMOVE_HEAD(obj->children, object_link); + + if (depth > 0) { + SLIST_FOREACH(child, obj->children, object_link) + { + error = libsysctl_internal_subtree(child, flags, + depth - 1); + if (error != 0) { + return (error); + } + } + } + + return (error); +} + + +/* API implementation */ + +int +libsysctl_nametoid(const char *name, size_t namelen, int *id, size_t *idlevel) +{ + int mib[2]; + int error = 0; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_NAME2OID; + + *idlevel *= sizeof(int); + error = sysctl(mib, 2, id, idlevel, name, namelen); + *idlevel /= sizeof(int); + + return (error); +} + + +int +libsysctl_desc(int *id, size_t idlevel, char *desc, size_t *desclen) +{ + int mib[CTL_MAXNAME]; + int error = 0; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_OBJECTDESCR; + memcpy(mib + 2, id, idlevel * sizeof(int)); + + error = sysctl(mib, idlevel+2, (void *)desc, desclen, NULL, 0); + + return (error); +} + + +int +libsysctl_name(int *id, size_t idlevel, char *name, size_t *namelen) +{ + int mib[CTL_MAXNAME]; + int error = 0; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_OBJECTNAME; + memcpy(mib + 2, id, idlevel * sizeof(int)); + + error = sysctl(mib, idlevel+2, (void *)name, namelen, NULL, 0); + + return (error); +} + + +int +libsysctl_label(int *id, size_t idlevel, char *label, size_t *labellen) +{ + int mib[CTL_MAXNAME]; + int error = 0; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_OBJECTLABEL; + memcpy(mib + 2, id, idlevel * sizeof(int)); + + error = sysctl(mib, idlevel+2, (void *)label, labellen, NULL, 0); + + return (error); +} + + +int +libsysctl_info(int *id, size_t idlevel, void *info, size_t *infosize) +{ + int mib[CTL_MAXNAME]; + int error = 0; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_OBJECTFMT; + memcpy(mib + 2, id, idlevel*sizeof(int)); + + error = sysctl(mib, idlevel+2, info, infosize, NULL, 0); + + return (error); +} + + +int +libsysctl_nextleaf(int *id, size_t idlevel, int *idnext, size_t *idnextlevel) +{ + int mib[CTL_MAXNAME]; + int error = 0; + size_t tmp_nextlevel; + + mib[0] = CTL_SYSCTLMIB; + mib[1] = MIB_NEXTOID; + memcpy(mib + 2, id, idlevel*sizeof(int)); + + tmp_nextlevel = *idnextlevel * sizeof(int); + + error = sysctl(mib, idlevel+2, idnext, &tmp_nextlevel, NULL, 0); + if (error == 0) { + *idnextlevel = tmp_nextlevel / sizeof(int); + } + + return (error); +} + + +int +libsysctl_nextnode(int *id, size_t idlevel, int *idnext, size_t *idnextlevel) +{ + int error = 0; + size_t i, minlen; + size_t tmp_nextlevel; + + /* it could be "id = idnext", then: */ + int previd[CTL_MAXNAME]; + size_t prevlevel = idlevel; + + memcpy(previd, id, idlevel * sizeof(int)); + + tmp_nextlevel = *idnextlevel * sizeof(int); + + error = libsysctl_nextleaf(id, idlevel, idnext, &tmp_nextlevel); + if (error != 0) { + return (error); + } + + *idnextlevel = tmp_nextlevel; + + /* + * avoid: id 5.6 -> next 5.6.4.8.2 + * we want: id 5.6 -> next 5.6.4 (just 1 level) + */ + if (*idnextlevel > prevlevel) { + *idnextlevel = prevlevel + 1; + } + + minlen = *idnextlevel < prevlevel ? *idnextlevel : prevlevel; + + for (i = 0; i < minlen; i++) { + if (previd[i] != idnext[i]) { + *idnextlevel = i+1; + break; + } + } + + return (error); +} + + +struct libsysctl_object * +libsysctl_object(int *id, size_t idlevel, unsigned int flags) +{ + struct libsysctl_object *obj = NULL; + size_t size = 0; + void *tmpinfo = NULL; + + obj = malloc(sizeof(struct libsysctl_object)); + if (obj == NULL) { + return (NULL); + } + + /* init new object */ + memset(obj, 0, sizeof(struct libsysctl_object)); + + /* id and idlevel are always set */ + obj->id = malloc(idlevel * sizeof(int)); + if (obj->id == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + memcpy(obj->id, id, idlevel * sizeof(int)); + obj->idlevel = idlevel; + + if (flags & LIBSYSCTL_FNAME) { + /* kernel returns false positive */ + if (LIBSYSCTL_NAMELEN(id, idlevel, &size) == 0) { + if ((obj->name = malloc(size)) == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + bzero(obj->name, size); + if (libsysctl_name(id, idlevel, obj->name, + &size) != 0) { + obj->name = NULL; + } + } + } + + + if (flags & LIBSYSCTL_FDESC) { + size = 0; + /* entry without descr could return "\0" or NULL */ + if (LIBSYSCTL_DESCLEN(id, idlevel, &size) == 0) { + if ((obj->desc = malloc(size)) == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + bzero(obj->desc, size); + if (libsysctl_desc(id, idlevel, obj->desc, + &size) != 0) { + obj->desc = NULL; + } + } + } + + + if (flags & LIBSYSCTL_FLABEL) { + size = 0; + if (LIBSYSCTL_LABELLEN(id, idlevel, &size) == 0) { + if ((obj->label = malloc(size)) == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + bzero(obj->label, size); + if (libsysctl_label(id, idlevel, obj->label, + &size) != 0) { + obj->label = NULL; + } + } + } + + + if ((flags & LIBSYSCTL_FFLAGS) || (flags & LIBSYSCTL_FFMT) || + (flags & LIBSYSCTL_FTYPE)) { + size = 0; + /* get info size because fmt is variable */ + if (libsysctl_info(id, idlevel, NULL, &size) == 0) { + tmpinfo = malloc(size); + if (tmpinfo == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + + if (libsysctl_info(id, idlevel, tmpinfo, &size) < 0) { + libsysctl_freeobject(obj); + free(tmpinfo); + return (NULL); + } + + if (flags & LIBSYSCTL_FFMT) { + obj->fmt = strndup(LIBSYSCTL_INFOFMT(tmpinfo), + size - sizeof(uint32_t)); + if (obj->fmt == NULL) { + libsysctl_freeobject(obj); + return (NULL); + } + } + + if (flags & LIBSYSCTL_FFLAGS) { + obj->flags = LIBSYSCTL_INFOFLAGS(tmpinfo); + } + + if (flags & LIBSYSCTL_FTYPE) { + obj->type = LIBSYSCTL_INFOTYPE(tmpinfo); + } + + free(tmpinfo); + } + } + + return (obj); +} + + +void +libsysctl_freeobject(struct libsysctl_object *object) +{ + if (object == NULL) { + return; + } + + free(object->id); + free(object->name); + free(object->desc); + free(object->label); + free(object); + object = NULL; +} + + +struct libsysctl_object_list * +libsysctl_filterlist(libsysctl_filterfunc_t *filterfunc, unsigned int flags) +{ + int id[CTL_MAXNAME], idnext[CTL_MAXNAME]; + size_t idlevel, idnextlevel; + struct libsysctl_object_list *list = NULL; + struct libsysctl_object *last, *new; + + list = malloc(sizeof(struct libsysctl_object_list)); + if (list == NULL) { + return (NULL); + } + + SLIST_INIT(list); + + id[0] = 0; + idlevel = 1; + + for (;;) { + if ((new = libsysctl_object(id, idlevel, flags)) == NULL) { + libsysctl_freelist(list); + return (NULL); + } + + if ((filterfunc == NULL) || (filterfunc(new) == 0)) { + if (SLIST_EMPTY(list)) { + SLIST_INSERT_HEAD(list, new, object_link); + } else { + SLIST_INSERT_AFTER(last, new, object_link); + } + + last = new; + } + + idnextlevel = CTL_MAXNAME; + if (libsysctl_nextnode(id, idlevel, idnext, &idnextlevel) < 0) { + break; + } + memcpy(id, idnext, idnextlevel * sizeof(int)); + idlevel = idnextlevel; + } + + return (list); +} + + +struct libsysctl_object_list * +libsysctl_grouplist(int *idstart, size_t idstartlen, unsigned int flags, + unsigned int depth) +{ + int id[CTL_MAXNAME], idnext[CTL_MAXNAME]; + size_t idlevel, idnextlevel; + struct libsysctl_object_list *list = NULL; + struct libsysctl_object *last, *new; + size_t i; + + list = malloc(sizeof(struct libsysctl_object_list)); + if (list == NULL) { + return (NULL); + } + + SLIST_INIT(list); + + memcpy(id, idstart, idstartlen * sizeof(int)); + idlevel = idstartlen; + + if ((new = libsysctl_object(id, idlevel, flags)) == NULL) { + free(list); + return (NULL); + } + + SLIST_INSERT_HEAD(list, new, object_link); + + last = new; + + + for (;;) { + idnextlevel = CTL_MAXNAME; + if (libsysctl_nextnode(id, idlevel, idnext, &idnextlevel) < 0) { + break; + } + memcpy(id, idnext, idnextlevel * sizeof(int)); + idlevel = idnextlevel; + + if (idlevel - idstartlen > depth) { + continue; + } + + if (idlevel < idstartlen) { + break; + } + + for (i = 0; i < idstartlen; i++) { + if (id[i] != idstart[i]) { + return (list); + } + } + + new = libsysctl_object(id, idlevel, flags); + if (new == NULL) { + libsysctl_freelist(list); + return (NULL); + } + SLIST_INSERT_AFTER(last, new, object_link); + last = new; + } + + return (list); +} + + +void +libsysctl_freelist(struct libsysctl_object_list *list) +{ + struct libsysctl_object *obj; + + if (list == NULL) { + return; + } + + while (!SLIST_EMPTY(list)) { + obj = SLIST_FIRST(list); + SLIST_REMOVE_HEAD(list, object_link); + libsysctl_freeobject(obj); + } + + free(list); + list = NULL; +} + + +struct libsysctl_object * +libsysctl_tree(int *id, size_t idlevel, unsigned int flags, + unsigned int max_depth) +{ + int error; + struct libsysctl_object *root = NULL; + + if ((root = libsysctl_object(id, idlevel, flags)) == NULL) { + return (NULL); + } + + if (max_depth < 1) { + return (root); + } + + error = libsysctl_internal_subtree(root, flags, max_depth - 1); + if (error != 0) { + libsysctl_freetree(root); + } + + return (root); +} + + +/* postorder visit */ +void +libsysctl_freetree(struct libsysctl_object *node) +{ + struct libsysctl_object *child; + + if (node == NULL) { + return; + } + + if (node->children != NULL) { + while (!SLIST_EMPTY(node->children)) { + child = SLIST_FIRST(node->children); + SLIST_REMOVE_HEAD(node->children, object_link); + libsysctl_freetree(child); + } + } + + + libsysctl_freeobject(node); + node = NULL; +}