On Fri, Sep 17, 2021 at 12:56 PM Andrii Nakryiko <andrii.nakryiko@xxxxxxxxx> wrote: > > On Fri, Sep 17, 2021 at 8:25 AM grantseltzer <grantseltzer@xxxxxxxxx> wrote: > > > > From: Grant Seltzer <grantseltzer@xxxxxxxxx> > > > > This adds comments above functions in libbpf.h which document > > their uses. These comments are of a format that doxygen and sphinx > > can pick up and render. These are rendered by libbpf.readthedocs.org > > > > These doc comments are for: > > - bpf_object__find_map_by_name() > > - bpf_map__fd() > > - bpf_map__is_internal() > > - libbpf_get_error() > > - libbpf_num_possible_cpus() > > > > Signed-off-by: Grant Seltzer <grantseltzer@xxxxxxxxx> > > --- > > tools/lib/bpf/libbpf.h | 58 ++++++++++++++++++++++++++++++++++++------ > > 1 file changed, 50 insertions(+), 8 deletions(-) > > > > diff --git a/tools/lib/bpf/libbpf.h b/tools/lib/bpf/libbpf.h > > index 2f6f0e15d1e7..27a5ebf56d19 100644 > > --- a/tools/lib/bpf/libbpf.h > > +++ b/tools/lib/bpf/libbpf.h > > @@ -478,9 +478,14 @@ struct bpf_map_def { > > unsigned int map_flags; > > }; > > > > -/* > > - * The 'struct bpf_map' in include/linux/bpf.h is internal to the kernel, > > - * so no need to worry about a name clash. > > +/** > > + * @brief **bpf_object__find_map_by_name()** returns a pointer to the > > this whole "a pointer" wording seems very low-level. It's clear that > it's a pointer just by looking at the function signature. Maybe let's > use a slightly higher-level terminology when talking about bpf_map, > bpf_program, bpf_object, etc. E.g., how about something like this: > > "... returns BPF map of the given name, if it exists within the passed > BPF object." No need to describe what happens if it doesn't exist > here, because we have @return section, which can be: > > "@return BPF map instance, if such map exists within BPF object; NULL, > otherwise. > > > + * specified bpf map in the bpf object if that map exists, and returns > > let's use BPF (all caps) consistently throughout documentation > > > + * NULL if not. It sets errno in case of error. > > + * @param obj bpf object > > + * @param name name of the bpf map > > + * @return the address of the map within the bpf object, or NULL if it > > + * does not exist > > */ > > LIBBPF_API struct bpf_map * > > bpf_object__find_map_by_name(const struct bpf_object *obj, const char *name); > > @@ -506,7 +511,15 @@ bpf_map__next(const struct bpf_map *map, const struct bpf_object *obj); > > LIBBPF_API struct bpf_map * > > bpf_map__prev(const struct bpf_map *map, const struct bpf_object *obj); > > > > -/* get/set map FD */ > > +/** > > + * @brief **bpf_map__fd()** gets the file descriptor of the passed > > + * bpf map > > + * @param map the bpf map instance > > + * @return the file descriptor or in case of an error, EINVAL > > or -EINVAL, not EINVAL > > > + * > > + * errno should be checked after this call, it will be EINVAL in > > + * case of error. > > this last part is misleading. -EINVAL is returned directly so you > don't have to check errno. Let's maybe drop this sentence altogether, > it's going to be extremely repetitive to specify that for each API. We > should have a separate section about libbpf's approach to returning > errors, for low-level/high-level int-returning APIs and > pointer-returning APIs. > > > + */ > > LIBBPF_API int bpf_map__fd(const struct bpf_map *map); > > LIBBPF_API int bpf_map__reuse_fd(struct bpf_map *map, int fd); > > /* get map definition */ > > @@ -547,6 +560,15 @@ LIBBPF_API int bpf_map__set_initial_value(struct bpf_map *map, > > const void *data, size_t size); > > LIBBPF_API const void *bpf_map__initial_value(struct bpf_map *map, size_t *psize); > > LIBBPF_API bool bpf_map__is_offload_neutral(const struct bpf_map *map); > > + > > +/** > > + * @brief **bpf_map__is_internal()** tells the caller whether or not > > + * the passed map is a special internal map > > let's expand a little bit: "a special map created by libbpf > automatically for things like global variables, __ksym externs, > Kconfig values, etc"? > > > + * @param map reference to the bpf_map > > + * @return true if the map is an internal map, false if not > > s/if not/otherwise/ > > > + * > > + * See the enum `libbpf_map_type` for listing of the types > > + */ > > LIBBPF_API bool bpf_map__is_internal(const struct bpf_map *map); > > LIBBPF_API int bpf_map__set_pin_path(struct bpf_map *map, const char *path); > > LIBBPF_API const char *bpf_map__get_pin_path(const struct bpf_map *map); > > @@ -558,6 +580,24 @@ LIBBPF_API int bpf_map__unpin(struct bpf_map *map, const char *path); > > LIBBPF_API int bpf_map__set_inner_map_fd(struct bpf_map *map, int fd); > > LIBBPF_API struct bpf_map *bpf_map__inner_map(struct bpf_map *map); > > > > +/** > > + * @brief **libbpf_get_error()** extracts the error code from the passed > > + * pointer > > + * @param ptr pointer returned from libbpf API function > > + * @return error code > > "or 0, if no error happened" > > > + * > > + * Many libbpf API functions which return pointers have logic to encode error > > + * codes as pointers, and do not return NULL. Meaning **libbpf_get_error()** > > + * should be used on the return value from these functions. Consult the > > + * individual functions documentation to verify if this logic applies. > > + * > > + * For these API functions, if `libbpf_set_strict_mode(LIBBPF_STRICT_CLEAN_PTRS)` > > + * is enabled, NULL is returned on error instead. > > + * > > + * If ptr == NULL, then errno should be already set by the failing > > nit: "*ptr* is NULL", it's not a code, it's docs for humans, so > probably better to use conversational language here. Also do > parameters have to be enclosed in *...*? Or it's just a typographic > convention? Purely convention. > > > Also, I think it's worth noting here that `libbpf_get_error()` has to > be called right after the API itself with no other intervening calls > that could modify errno. You might get away with this in non-strict > mode, but in strict mode when pointer is NULL, the only way to get > error code is through errno variable (which is what libbpf_get_error() > is doing), but that assumes that errno is preserved. So probably worth > stating "Use libbpf_get_error() to extract error immediately after > calling an API function, with no intervening calls that could clobber > `errno` variable." or something along those lines? > > > + * API, because libbpf never returns NULL on success and it now always > > + * sets errno on error. > > + */ > > LIBBPF_API long libbpf_get_error(const void *ptr); > > > > struct bpf_prog_load_attr { > > @@ -822,9 +862,12 @@ bpf_program__bpil_addr_to_offs(struct bpf_prog_info_linear *info_linear); > > LIBBPF_API void > > bpf_program__bpil_offs_to_addr(struct bpf_prog_info_linear *info_linear); > > > > -/* > > - * A helper function to get the number of possible CPUs before looking up > > - * per-CPU maps. Negative errno is returned on failure. > > +/** > > + * @brief **libbpf_num_possible_cpus()** is helper function to get the > > nit: is *a* helper function > > > + * number of possible CPUs before looking up per-CPU maps. > > "before looking up per-CPU maps" is too specific, it's not the only > case. So let's keep it generic. It's the theoretically possible number > of CPUs that a host kernel supports and expects. > > > + * @return number of possible CPUs > > + * > > + * Negative errno is returned on failure. > > errno is misleading here, is this about the thread-local errno > variable or an error code? Let's use "error code" consistently > throughout the docs. > > > * > > * Example usage: > > * > > @@ -834,7 +877,6 @@ bpf_program__bpil_offs_to_addr(struct bpf_prog_info_linear *info_linear); > > * } > > * long values[ncpus]; > > * bpf_map_lookup_elem(per_cpu_map_fd, key, values); > > - * > > */ > > LIBBPF_API int libbpf_num_possible_cpus(void); > > > > -- > > 2.31.1 > >