18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci// Copyright (C) 2018 Joe Lawrence <joe.lawrence@redhat.com>
38c2ecf20Sopenharmony_ci
48c2ecf20Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
58c2ecf20Sopenharmony_ci
68c2ecf20Sopenharmony_ci#include <linux/module.h>
78c2ecf20Sopenharmony_ci#include <linux/kernel.h>
88c2ecf20Sopenharmony_ci#include <linux/livepatch.h>
98c2ecf20Sopenharmony_ci
108c2ecf20Sopenharmony_cistatic int replace;
118c2ecf20Sopenharmony_cimodule_param(replace, int, 0644);
128c2ecf20Sopenharmony_ciMODULE_PARM_DESC(replace, "replace (default=0)");
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_cistatic const char *const module_state[] = {
158c2ecf20Sopenharmony_ci	[MODULE_STATE_LIVE]	= "[MODULE_STATE_LIVE] Normal state",
168c2ecf20Sopenharmony_ci	[MODULE_STATE_COMING]	= "[MODULE_STATE_COMING] Full formed, running module_init",
178c2ecf20Sopenharmony_ci	[MODULE_STATE_GOING]	= "[MODULE_STATE_GOING] Going away",
188c2ecf20Sopenharmony_ci	[MODULE_STATE_UNFORMED]	= "[MODULE_STATE_UNFORMED] Still setting it up",
198c2ecf20Sopenharmony_ci};
208c2ecf20Sopenharmony_ci
218c2ecf20Sopenharmony_cistatic void callback_info(const char *callback, struct klp_object *obj)
228c2ecf20Sopenharmony_ci{
238c2ecf20Sopenharmony_ci	if (obj->mod)
248c2ecf20Sopenharmony_ci		pr_info("%s: %s -> %s\n", callback, obj->mod->name,
258c2ecf20Sopenharmony_ci			module_state[obj->mod->state]);
268c2ecf20Sopenharmony_ci	else
278c2ecf20Sopenharmony_ci		pr_info("%s: vmlinux\n", callback);
288c2ecf20Sopenharmony_ci}
298c2ecf20Sopenharmony_ci
308c2ecf20Sopenharmony_ci/* Executed on object patching (ie, patch enablement) */
318c2ecf20Sopenharmony_cistatic int pre_patch_callback(struct klp_object *obj)
328c2ecf20Sopenharmony_ci{
338c2ecf20Sopenharmony_ci	callback_info(__func__, obj);
348c2ecf20Sopenharmony_ci	return 0;
358c2ecf20Sopenharmony_ci}
368c2ecf20Sopenharmony_ci
378c2ecf20Sopenharmony_ci/* Executed on object unpatching (ie, patch disablement) */
388c2ecf20Sopenharmony_cistatic void post_patch_callback(struct klp_object *obj)
398c2ecf20Sopenharmony_ci{
408c2ecf20Sopenharmony_ci	callback_info(__func__, obj);
418c2ecf20Sopenharmony_ci}
428c2ecf20Sopenharmony_ci
438c2ecf20Sopenharmony_ci/* Executed on object unpatching (ie, patch disablement) */
448c2ecf20Sopenharmony_cistatic void pre_unpatch_callback(struct klp_object *obj)
458c2ecf20Sopenharmony_ci{
468c2ecf20Sopenharmony_ci	callback_info(__func__, obj);
478c2ecf20Sopenharmony_ci}
488c2ecf20Sopenharmony_ci
498c2ecf20Sopenharmony_ci/* Executed on object unpatching (ie, patch disablement) */
508c2ecf20Sopenharmony_cistatic void post_unpatch_callback(struct klp_object *obj)
518c2ecf20Sopenharmony_ci{
528c2ecf20Sopenharmony_ci	callback_info(__func__, obj);
538c2ecf20Sopenharmony_ci}
548c2ecf20Sopenharmony_ci
558c2ecf20Sopenharmony_cistatic struct klp_func no_funcs[] = {
568c2ecf20Sopenharmony_ci	{ }
578c2ecf20Sopenharmony_ci};
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistatic struct klp_object objs[] = {
608c2ecf20Sopenharmony_ci	{
618c2ecf20Sopenharmony_ci		.name = NULL,	/* vmlinux */
628c2ecf20Sopenharmony_ci		.funcs = no_funcs,
638c2ecf20Sopenharmony_ci		.callbacks = {
648c2ecf20Sopenharmony_ci			.pre_patch = pre_patch_callback,
658c2ecf20Sopenharmony_ci			.post_patch = post_patch_callback,
668c2ecf20Sopenharmony_ci			.pre_unpatch = pre_unpatch_callback,
678c2ecf20Sopenharmony_ci			.post_unpatch = post_unpatch_callback,
688c2ecf20Sopenharmony_ci		},
698c2ecf20Sopenharmony_ci	}, { }
708c2ecf20Sopenharmony_ci};
718c2ecf20Sopenharmony_ci
728c2ecf20Sopenharmony_cistatic struct klp_patch patch = {
738c2ecf20Sopenharmony_ci	.mod = THIS_MODULE,
748c2ecf20Sopenharmony_ci	.objs = objs,
758c2ecf20Sopenharmony_ci	/* set .replace in the init function below for demo purposes */
768c2ecf20Sopenharmony_ci};
778c2ecf20Sopenharmony_ci
788c2ecf20Sopenharmony_cistatic int test_klp_callbacks_demo2_init(void)
798c2ecf20Sopenharmony_ci{
808c2ecf20Sopenharmony_ci	patch.replace = replace;
818c2ecf20Sopenharmony_ci	return klp_enable_patch(&patch);
828c2ecf20Sopenharmony_ci}
838c2ecf20Sopenharmony_ci
848c2ecf20Sopenharmony_cistatic void test_klp_callbacks_demo2_exit(void)
858c2ecf20Sopenharmony_ci{
868c2ecf20Sopenharmony_ci}
878c2ecf20Sopenharmony_ci
888c2ecf20Sopenharmony_cimodule_init(test_klp_callbacks_demo2_init);
898c2ecf20Sopenharmony_cimodule_exit(test_klp_callbacks_demo2_exit);
908c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
918c2ecf20Sopenharmony_ciMODULE_INFO(livepatch, "Y");
928c2ecf20Sopenharmony_ciMODULE_AUTHOR("Joe Lawrence <joe.lawrence@redhat.com>");
938c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Livepatch test: livepatch demo2");
94