mirror of
https://github.com/torvalds/linux.git
synced 2026-05-12 16:18:45 +02:00
In preparation for removing the strlcat API[1], replace the char *pp_buf
with a struct seq_buf, which tracks the current write position and
remaining space internally. This allows for:
- Direct use of seq_buf_printf() in place of snprintf()+strlcat()
pairs, eliminating local tmp buffers throughout.
- Adjacent strlcat() calls that build strings piece-by-piece
(e.g., strlcat("["); strlcat(name); strlcat("]")) to be collapsed
into single seq_buf_printf() calls.
- Simpler call sites: seq_buf_puts() takes only the buffer and string,
with no need to pass PAGE_SIZE at every call.
The backing buffer allocation is unchanged (__get_free_page), and the
output path uses seq_buf_str() to NUL-terminate before passing to
printk().
Link: https://github.com/KSPP/linux/issues/370 [1]
Cc: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Cc: Josh Law <objecting@objecting.org>
Signed-off-by: Kees Cook <kees@kernel.org>
Reviewed-by: Josh Law <objecting@objecting.org>
Link: https://patch.msgid.link/20260321004840.work.670-kees@kernel.org
Signed-off-by: Jens Axboe <axboe@kernel.dk>
109 lines
2.5 KiB
C
109 lines
2.5 KiB
C
// 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;
|
|
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));
|
|
|
|
seq_buf_printf(&state->pp_buf, "(%s)", info->volname);
|
|
}
|
|
|
|
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++;
|
|
}
|
|
|
|
seq_buf_puts(&state->pp_buf, "\n");
|
|
|
|
return 1;
|
|
}
|