Skip to content

Commit 8e068a2

Browse files
soleenakpm00
authored andcommitted
kho: update FDT dynamically for subtree addition/removal
Currently, sub-FDTs were tracked in a list (kho_out.sub_fdts) and the final FDT is constructed entirely from scratch during kho_finalize(). We can maintain the FDT dynamically: 1. Initialize a valid, empty FDT in kho_init(). 2. Use fdt_add_subnode and fdt_setprop in kho_add_subtree to update the FDT immediately when a subsystem registers. 3. Use fdt_del_node in kho_remove_subtree to remove entries. This removes the need for the intermediate sub_fdts list and the reconstruction logic in kho_finalize(). kho_finalize() now only needs to trigger memory map serialization. Link: https://lkml.kernel.org/r/20251114190002.3311679-11-pasha.tatashin@soleen.com Signed-off-by: Pasha Tatashin <pasha.tatashin@soleen.com> Reviewed-by: Mike Rapoport (Microsoft) <rppt@kernel.org> Reviewed-by: Pratyush Yadav <pratyush@kernel.org> Cc: Alexander Graf <graf@amazon.com> Cc: Arnd Bergmann <arnd@arndb.de> Cc: Baoquan He <bhe@redhat.com> Cc: Coiby Xu <coxu@redhat.com> Cc: Dave Vasilevsky <dave@vasilevsky.ca> Cc: Eric Biggers <ebiggers@google.com> Cc: Kees Cook <kees@kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
1 parent 9a4301f commit 8e068a2

1 file changed

Lines changed: 69 additions & 75 deletions

File tree

kernel/liveupdate/kexec_handover.c

Lines changed: 69 additions & 75 deletions
Original file line numberDiff line numberDiff line change
@@ -104,20 +104,11 @@ struct kho_mem_track {
104104

105105
struct khoser_mem_chunk;
106106

107-
struct kho_sub_fdt {
108-
struct list_head l;
109-
const char *name;
110-
void *fdt;
111-
};
112-
113107
struct kho_out {
114108
void *fdt;
115109
bool finalized;
116110
struct mutex lock; /* protects KHO FDT finalization */
117111

118-
struct list_head sub_fdts;
119-
struct mutex fdts_lock;
120-
121112
struct kho_mem_track track;
122113
struct kho_debugfs dbg;
123114
};
@@ -127,8 +118,6 @@ static struct kho_out kho_out = {
127118
.track = {
128119
.orders = XARRAY_INIT(kho_out.track.orders, 0),
129120
},
130-
.sub_fdts = LIST_HEAD_INIT(kho_out.sub_fdts),
131-
.fdts_lock = __MUTEX_INITIALIZER(kho_out.fdts_lock),
132121
.finalized = false,
133122
};
134123

@@ -725,37 +714,67 @@ static void __init kho_reserve_scratch(void)
725714
*/
726715
int kho_add_subtree(const char *name, void *fdt)
727716
{
728-
struct kho_sub_fdt *sub_fdt;
717+
phys_addr_t phys = virt_to_phys(fdt);
718+
void *root_fdt = kho_out.fdt;
719+
int err = -ENOMEM;
720+
int off, fdt_err;
729721

730-
sub_fdt = kmalloc(sizeof(*sub_fdt), GFP_KERNEL);
731-
if (!sub_fdt)
732-
return -ENOMEM;
722+
guard(mutex)(&kho_out.lock);
723+
724+
fdt_err = fdt_open_into(root_fdt, root_fdt, PAGE_SIZE);
725+
if (fdt_err < 0)
726+
return err;
733727

734-
INIT_LIST_HEAD(&sub_fdt->l);
735-
sub_fdt->name = name;
736-
sub_fdt->fdt = fdt;
728+
off = fdt_add_subnode(root_fdt, 0, name);
729+
if (off < 0) {
730+
if (off == -FDT_ERR_EXISTS)
731+
err = -EEXIST;
732+
goto out_pack;
733+
}
734+
735+
err = fdt_setprop(root_fdt, off, PROP_SUB_FDT, &phys, sizeof(phys));
736+
if (err < 0)
737+
goto out_pack;
737738

738-
guard(mutex)(&kho_out.fdts_lock);
739-
list_add_tail(&sub_fdt->l, &kho_out.sub_fdts);
740739
WARN_ON_ONCE(kho_debugfs_fdt_add(&kho_out.dbg, name, fdt, false));
741740

742-
return 0;
741+
out_pack:
742+
fdt_pack(root_fdt);
743+
744+
return err;
743745
}
744746
EXPORT_SYMBOL_GPL(kho_add_subtree);
745747

746748
void kho_remove_subtree(void *fdt)
747749
{
748-
struct kho_sub_fdt *sub_fdt;
750+
phys_addr_t target_phys = virt_to_phys(fdt);
751+
void *root_fdt = kho_out.fdt;
752+
int off;
753+
int err;
754+
755+
guard(mutex)(&kho_out.lock);
756+
757+
err = fdt_open_into(root_fdt, root_fdt, PAGE_SIZE);
758+
if (err < 0)
759+
return;
760+
761+
for (off = fdt_first_subnode(root_fdt, 0); off >= 0;
762+
off = fdt_next_subnode(root_fdt, off)) {
763+
const u64 *val;
764+
int len;
765+
766+
val = fdt_getprop(root_fdt, off, PROP_SUB_FDT, &len);
767+
if (!val || len != sizeof(phys_addr_t))
768+
continue;
749769

750-
guard(mutex)(&kho_out.fdts_lock);
751-
list_for_each_entry(sub_fdt, &kho_out.sub_fdts, l) {
752-
if (sub_fdt->fdt == fdt) {
753-
list_del(&sub_fdt->l);
754-
kfree(sub_fdt);
770+
if ((phys_addr_t)*val == target_phys) {
771+
fdt_del_node(root_fdt, off);
755772
kho_debugfs_fdt_remove(&kho_out.dbg, fdt);
756773
break;
757774
}
758775
}
776+
777+
fdt_pack(root_fdt);
759778
}
760779
EXPORT_SYMBOL_GPL(kho_remove_subtree);
761780

@@ -1232,48 +1251,6 @@ void kho_restore_free(void *mem)
12321251
}
12331252
EXPORT_SYMBOL_GPL(kho_restore_free);
12341253

1235-
static int __kho_finalize(void)
1236-
{
1237-
void *root = kho_out.fdt;
1238-
struct kho_sub_fdt *fdt;
1239-
u64 empty_mem_map = 0;
1240-
int err;
1241-
1242-
err = fdt_create(root, PAGE_SIZE);
1243-
err |= fdt_finish_reservemap(root);
1244-
err |= fdt_begin_node(root, "");
1245-
err |= fdt_property_string(root, "compatible", KHO_FDT_COMPATIBLE);
1246-
err |= fdt_property(root, PROP_PRESERVED_MEMORY_MAP, &empty_mem_map,
1247-
sizeof(empty_mem_map));
1248-
if (err)
1249-
goto err_exit;
1250-
1251-
mutex_lock(&kho_out.fdts_lock);
1252-
list_for_each_entry(fdt, &kho_out.sub_fdts, l) {
1253-
phys_addr_t phys = virt_to_phys(fdt->fdt);
1254-
1255-
err |= fdt_begin_node(root, fdt->name);
1256-
err |= fdt_property(root, PROP_SUB_FDT, &phys, sizeof(phys));
1257-
err |= fdt_end_node(root);
1258-
}
1259-
mutex_unlock(&kho_out.fdts_lock);
1260-
1261-
err |= fdt_end_node(root);
1262-
err |= fdt_finish(root);
1263-
if (err)
1264-
goto err_exit;
1265-
1266-
err = kho_mem_serialize(&kho_out);
1267-
if (err)
1268-
goto err_exit;
1269-
1270-
return 0;
1271-
1272-
err_exit:
1273-
pr_err("Failed to convert KHO state tree: %d\n", err);
1274-
return err;
1275-
}
1276-
12771254
int kho_finalize(void)
12781255
{
12791256
int ret;
@@ -1282,12 +1259,7 @@ int kho_finalize(void)
12821259
return -EOPNOTSUPP;
12831260

12841261
guard(mutex)(&kho_out.lock);
1285-
if (kho_out.finalized) {
1286-
kho_update_memory_map(NULL);
1287-
kho_out.finalized = false;
1288-
}
1289-
1290-
ret = __kho_finalize();
1262+
ret = kho_mem_serialize(&kho_out);
12911263
if (ret)
12921264
return ret;
12931265

@@ -1372,6 +1344,24 @@ int kho_retrieve_subtree(const char *name, phys_addr_t *phys)
13721344
}
13731345
EXPORT_SYMBOL_GPL(kho_retrieve_subtree);
13741346

1347+
static __init int kho_out_fdt_setup(void)
1348+
{
1349+
void *root = kho_out.fdt;
1350+
u64 empty_mem_map = 0;
1351+
int err;
1352+
1353+
err = fdt_create(root, PAGE_SIZE);
1354+
err |= fdt_finish_reservemap(root);
1355+
err |= fdt_begin_node(root, "");
1356+
err |= fdt_property_string(root, "compatible", KHO_FDT_COMPATIBLE);
1357+
err |= fdt_property(root, PROP_PRESERVED_MEMORY_MAP, &empty_mem_map,
1358+
sizeof(empty_mem_map));
1359+
err |= fdt_end_node(root);
1360+
err |= fdt_finish(root);
1361+
1362+
return err;
1363+
}
1364+
13751365
static __init int kho_init(void)
13761366
{
13771367
const void *fdt = kho_get_fdt();
@@ -1394,6 +1384,10 @@ static __init int kho_init(void)
13941384
if (err)
13951385
goto err_free_fdt;
13961386

1387+
err = kho_out_fdt_setup();
1388+
if (err)
1389+
goto err_free_fdt;
1390+
13971391
if (fdt) {
13981392
kho_in_debugfs_init(&kho_in.dbg, fdt);
13991393
return 0;

0 commit comments

Comments
 (0)