| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- // SPDX-License-Identifier: GPL-2.0
- #include <linux/blkdev.h>
- #include <linux/major.h>
- #include <linux/of.h>
- #include <linux/string.h>
- #include "check.h"
- static int validate_of_partition(struct device_node *np, int slot)
- {
- u64 offset, size;
- int len;
- const __be32 *reg = of_get_property(np, "reg", &len);
- int a_cells = of_n_addr_cells(np);
- int s_cells = of_n_size_cells(np);
- /* Make sure reg len match the expected addr and size cells */
- if (len / sizeof(*reg) != a_cells + s_cells)
- return -EINVAL;
- /* Validate offset conversion from bytes to sectors */
- offset = of_read_number(reg, a_cells);
- if (offset % SECTOR_SIZE)
- return -EINVAL;
- /* Validate size conversion from bytes to sectors */
- size = of_read_number(reg + a_cells, s_cells);
- if (!size || size % SECTOR_SIZE)
- return -EINVAL;
- return 0;
- }
- static void add_of_partition(struct parsed_partitions *state, int slot,
- struct device_node *np)
- {
- struct partition_meta_info *info;
- char tmp[sizeof(info->volname) + 4];
- const char *partname;
- int len;
- const __be32 *reg = of_get_property(np, "reg", &len);
- int a_cells = of_n_addr_cells(np);
- int s_cells = of_n_size_cells(np);
- /* Convert bytes to sector size */
- u64 offset = of_read_number(reg, a_cells) / SECTOR_SIZE;
- u64 size = of_read_number(reg + a_cells, s_cells) / SECTOR_SIZE;
- put_partition(state, slot, offset, size);
- if (of_property_read_bool(np, "read-only"))
- state->parts[slot].flags |= ADDPART_FLAG_READONLY;
- /*
- * Follow MTD label logic, search for label property,
- * fallback to node name if not found.
- */
- info = &state->parts[slot].info;
- partname = of_get_property(np, "label", &len);
- if (!partname)
- partname = of_get_property(np, "name", &len);
- strscpy(info->volname, partname, sizeof(info->volname));
- snprintf(tmp, sizeof(tmp), "(%s)", info->volname);
- strlcat(state->pp_buf, tmp, PAGE_SIZE);
- }
- int of_partition(struct parsed_partitions *state)
- {
- struct device *ddev = disk_to_dev(state->disk);
- struct device_node *np;
- int slot;
- struct device_node *partitions_np = of_node_get(ddev->of_node);
- if (!partitions_np ||
- !of_device_is_compatible(partitions_np, "fixed-partitions"))
- return 0;
- slot = 1;
- /* Validate parition offset and size */
- for_each_child_of_node(partitions_np, np) {
- if (validate_of_partition(np, slot)) {
- of_node_put(np);
- of_node_put(partitions_np);
- return -1;
- }
- slot++;
- }
- slot = 1;
- for_each_child_of_node(partitions_np, np) {
- if (slot >= state->limit) {
- of_node_put(np);
- break;
- }
- add_of_partition(state, slot, np);
- slot++;
- }
- strlcat(state->pp_buf, "\n", PAGE_SIZE);
- return 1;
- }
|