// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2015 Masahiro Yamada <yamada.masahiro@socionext.com> * * Based on the original work in Linux by * Copyright (c) 2006 SUSE Linux Products GmbH * Copyright (c) 2006 Tejun Heo <teheo@suse.de> */ #define LOG_CATEGORY LOGC_DEVRES #include <common.h> #include <log.h> #include <malloc.h> #include <linux/compat.h> #include <linux/kernel.h> #include <linux/list.h> #include <dm/device.h> #include <dm/devres.h> #include <dm/root.h> #include <dm/util.h> /** enum devres_phase - Shows where resource was allocated * * DEVRES_PHASE_BIND: In the bind() method * DEVRES_PHASE_OFDATA: In the ofdata_to_platdata() method * DEVRES_PHASE_PROBE: In the probe() method */ enum devres_phase { DEVRES_PHASE_BIND, DEVRES_PHASE_OFDATA, DEVRES_PHASE_PROBE, }; /** * struct devres - Bookkeeping info for managed device resource * @entry: List to associate this structure with a device * @release: Callback invoked when this resource is released * @probe: Show where this resource was allocated * @name: Name of release function * @size: Size of resource data * @data: Resource data */ struct devres { struct list_head entry; dr_release_t release; enum devres_phase phase; #ifdef CONFIG_DEBUG_DEVRES const char *name; size_t size; #endif unsigned long long data[]; }; #ifdef CONFIG_DEBUG_DEVRES static void set_node_dbginfo(struct devres *dr, const char *name, size_t size) { dr->name = name; dr->size = size; } static void devres_log(struct udevice *dev, struct devres *dr, const char *op) { log_debug("%s: DEVRES %3s %p %s (%lu bytes)\n", dev->name, op, dr, dr->name, (unsigned long)dr->size); } #else /* CONFIG_DEBUG_DEVRES */ #define set_node_dbginfo(dr, n, s) do {} while (0) #define devres_log(dev, dr, op) do {} while (0) #endif #if CONFIG_DEBUG_DEVRES void *__devres_alloc(dr_release_t release, size_t size, gfp_t gfp, const char *name) #else void *_devres_alloc(dr_release_t release, size_t size, gfp_t gfp) #endif { size_t tot_size = sizeof(struct devres) + size; struct devres *dr; dr = kmalloc(tot_size, gfp); if (unlikely(!dr)) return NULL; INIT_LIST_HEAD(&dr->entry); dr->release = release; set_node_dbginfo(dr, name, size); return dr->data; } void devres_free(void *res) { if (res) { struct devres *dr = container_of(res, struct devres, data); assert_noisy(list_empty(&dr->entry)); kfree(dr); } } void devres_add(struct udevice *dev, void *res) { struct devres *dr = container_of(res, struct devres, data); devres_log(dev, dr, "ADD"); assert_noisy(list_empty(&dr->entry)); if (dev->flags & DM_FLAG_PLATDATA_VALID) dr->phase = DEVRES_PHASE_PROBE; else if (dev->flags & DM_FLAG_BOUND) dr->phase = DEVRES_PHASE_OFDATA; else dr->phase = DEVRES_PHASE_BIND; list_add_tail(&dr->entry, &dev->devres_head); } void *devres_find(struct udevice *dev, dr_release_t release, dr_match_t match, void *match_data) { struct devres *dr; list_for_each_entry_reverse(dr, &dev->devres_head, entry) { if (dr->release != release) continue; if (match && !match(dev, dr->data, match_data)) continue; return dr->data; } return NULL; } void *devres_get(struct udevice *dev, void *new_res, dr_match_t match, void *match_data) { struct devres *new_dr = container_of(new_res, struct devres, data); void *res; res = devres_find(dev, new_dr->release, match, match_data); if (!res) { devres_add(dev, new_res); res = new_res; new_res = NULL; } devres_free(new_res); return res; } void *devres_remove(struct udevice *dev, dr_release_t release, dr_match_t match, void *match_data) { void *res; res = devres_find(dev, release, match, match_data); if (res) { struct devres *dr = container_of(res, struct devres, data); list_del_init(&dr->entry); devres_log(dev, dr, "REM"); } return res; } int devres_destroy(struct udevice *dev, dr_release_t release, dr_match_t match, void *match_data) { void *res; res = devres_remove(dev, release, match, match_data); if (unlikely(!res)) return -ENOENT; devres_free(res); return 0; } int devres_release(struct udevice *dev, dr_release_t release, dr_match_t match, void *match_data) { void *res; res = devres_remove(dev, release, match, match_data); if (unlikely(!res)) return -ENOENT; (*release)(dev, res); devres_free(res); return 0; } static void release_nodes(struct udevice *dev, struct list_head *head, bool probe_and_ofdata_only) { struct devres *dr, *tmp; list_for_each_entry_safe_reverse(dr, tmp, head, entry) { if (probe_and_ofdata_only && dr->phase == DEVRES_PHASE_BIND) break; devres_log(dev, dr, "REL"); dr->release(dev, dr->data); list_del(&dr->entry); kfree(dr); } } void devres_release_probe(struct udevice *dev) { release_nodes(dev, &dev->devres_head, true); } void devres_release_all(struct udevice *dev) { release_nodes(dev, &dev->devres_head, false); } #ifdef CONFIG_DEBUG_DEVRES static char *const devres_phase_name[] = {"BIND", "OFDATA", "PROBE"}; static void dump_resources(struct udevice *dev, int depth) { struct devres *dr; struct udevice *child; printf("- %s\n", dev->name); list_for_each_entry(dr, &dev->devres_head, entry) printf(" %p (%lu byte) %s %s\n", dr, (unsigned long)dr->size, dr->name, devres_phase_name[dr->phase]); list_for_each_entry(child, &dev->child_head, sibling_node) dump_resources(child, depth + 1); } void dm_dump_devres(void) { struct udevice *root; root = dm_root(); if (root) dump_resources(root, 0); } void devres_get_stats(const struct udevice *dev, struct devres_stats *stats) { struct devres *dr; stats->allocs = 0; stats->total_size = 0; list_for_each_entry(dr, &dev->devres_head, entry) { stats->allocs++; stats->total_size += dr->size; } } #endif /* * Managed kmalloc/kfree */ static void devm_kmalloc_release(struct udevice *dev, void *res) { /* noop */ } static int devm_kmalloc_match(struct udevice *dev, void *res, void *data) { return res == data; } void *devm_kmalloc(struct udevice *dev, size_t size, gfp_t gfp) { void *data; data = _devres_alloc(devm_kmalloc_release, size, gfp); if (unlikely(!data)) return NULL; devres_add(dev, data); return data; } void devm_kfree(struct udevice *dev, void *p) { int rc; rc = devres_destroy(dev, devm_kmalloc_release, devm_kmalloc_match, p); assert_noisy(!rc); }