1======================
2(Un)patching Callbacks
3======================
4
5Livepatch (un)patch-callbacks provide a mechanism for livepatch modules
6to execute callback functions when a kernel object is (un)patched.  They
7can be considered a "power feature" that extends livepatching abilities
8to include:
9
10  - Safe updates to global data
11
12  - "Patches" to init and probe functions
13
14  - Patching otherwise unpatchable code (i.e. assembly)
15
16In most cases, (un)patch callbacks will need to be used in conjunction
17with memory barriers and kernel synchronization primitives, like
18mutexes/spinlocks, or even stop_machine(), to avoid concurrency issues.
19
20Callbacks differ from existing kernel facilities:
21
22  - Module init/exit code doesn't run when disabling and re-enabling a
23    patch.
24
25  - A module notifier can't stop a to-be-patched module from loading.
26
27Callbacks are part of the klp_object structure and their implementation
28is specific to that klp_object.  Other livepatch objects may or may not
29be patched, irrespective of the target klp_object's current state.
30
31Callbacks can be registered for the following livepatch actions:
32
33  * Pre-patch    - before a klp_object is patched
34
35  * Post-patch   - after a klp_object has been patched and is active
36                   across all tasks
37
38  * Pre-unpatch  - before a klp_object is unpatched (ie, patched code is
39                   active), used to clean up post-patch callback
40                   resources
41
42  * Post-unpatch - after a klp_object has been patched, all code has
43                   been restored and no tasks are running patched code,
44                   used to cleanup pre-patch callback resources
45
46Each callback is optional, omitting one does not preclude specifying any
47other.  However, the livepatching core executes the handlers in
48symmetry: pre-patch callbacks have a post-unpatch counterpart and
49post-patch callbacks have a pre-unpatch counterpart.  An unpatch
50callback will only be executed if its corresponding patch callback was
51executed.  Typical use cases pair a patch handler that acquires and
52configures resources with an unpatch handler tears down and releases
53those same resources.
54
55A callback is only executed if its host klp_object is loaded.  For
56in-kernel vmlinux targets, this means that callbacks will always execute
57when a livepatch is enabled/disabled.  For patch target kernel modules,
58callbacks will only execute if the target module is loaded.  When a
59module target is (un)loaded, its callbacks will execute only if the
60livepatch module is enabled.
61
62The pre-patch callback, if specified, is expected to return a status
63code (0 for success, -ERRNO on error).  An error status code indicates
64to the livepatching core that patching of the current klp_object is not
65safe and to stop the current patching request.  (When no pre-patch
66callback is provided, the transition is assumed to be safe.)  If a
67pre-patch callback returns failure, the kernel's module loader will:
68
69  - Refuse to load a livepatch, if the livepatch is loaded after
70    targeted code.
71
72    or:
73
74  - Refuse to load a module, if the livepatch was already successfully
75    loaded.
76
77No post-patch, pre-unpatch, or post-unpatch callbacks will be executed
78for a given klp_object if the object failed to patch, due to a failed
79pre_patch callback or for any other reason.
80
81If a patch transition is reversed, no pre-unpatch handlers will be run
82(this follows the previously mentioned symmetry -- pre-unpatch callbacks
83will only occur if their corresponding post-patch callback executed).
84
85If the object did successfully patch, but the patch transition never
86started for some reason (e.g., if another object failed to patch),
87only the post-unpatch callback will be called.
88
89
90Example Use-cases
91=================
92
93Update global data
94------------------
95
96A pre-patch callback can be useful to update a global variable.  For
97example, 75ff39ccc1bd ("tcp: make challenge acks less predictable")
98changes a global sysctl, as well as patches the tcp_send_challenge_ack()
99function.
100
101In this case, if we're being super paranoid, it might make sense to
102patch the data *after* patching is complete with a post-patch callback,
103so that tcp_send_challenge_ack() could first be changed to read
104sysctl_tcp_challenge_ack_limit with READ_ONCE.
105
106
107Support __init and probe function patches
108-----------------------------------------
109
110Although __init and probe functions are not directly livepatch-able, it
111may be possible to implement similar updates via pre/post-patch
112callbacks.
113
11448900cb6af42 ("virtio-net: drop NETIF_F_FRAGLIST") change the way that
115virtnet_probe() initialized its driver's net_device features.  A
116pre/post-patch callback could iterate over all such devices, making a
117similar change to their hw_features value.  (Client functions of the
118value may need to be updated accordingly.)
119
120
121Test cases
122==========
123
124What follows is not an exhaustive test suite of every possible livepatch
125pre/post-(un)patch combination, but a selection that demonstrates a few
126important concepts.  Each test case uses the kernel modules located in
127the samples/livepatch/ and assumes that no livepatches are loaded at the
128beginning of the test.
129
130
131Test 1
132------
133
134Test a combination of loading a kernel module and a livepatch that
135patches a function in the first module.  (Un)load the target module
136before the livepatch module:
137
138- load target module
139- load livepatch
140- disable livepatch
141- unload target module
142- unload livepatch
143
144First load a target module:
145
146  % insmod samples/livepatch/livepatch-callbacks-mod.ko
147  [   34.475708] livepatch_callbacks_mod: livepatch_callbacks_mod_init
148
149On livepatch enable, before the livepatch transition starts, pre-patch
150callbacks are executed for vmlinux and livepatch_callbacks_mod (those
151klp_objects currently loaded).  After klp_objects are patched according
152to the klp_patch, their post-patch callbacks run and the transition
153completes:
154
155  % insmod samples/livepatch/livepatch-callbacks-demo.ko
156  [   36.503719] livepatch: enabling patch 'livepatch_callbacks_demo'
157  [   36.504213] livepatch: 'livepatch_callbacks_demo': initializing patching transition
158  [   36.504238] livepatch_callbacks_demo: pre_patch_callback: vmlinux
159  [   36.504721] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
160  [   36.505849] livepatch: 'livepatch_callbacks_demo': starting patching transition
161  [   37.727133] livepatch: 'livepatch_callbacks_demo': completing patching transition
162  [   37.727232] livepatch_callbacks_demo: post_patch_callback: vmlinux
163  [   37.727860] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
164  [   37.728792] livepatch: 'livepatch_callbacks_demo': patching complete
165
166Similarly, on livepatch disable, pre-patch callbacks run before the
167unpatching transition starts.  klp_objects are reverted, post-patch
168callbacks execute and the transition completes:
169
170  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
171  [   38.510209] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
172  [   38.510234] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
173  [   38.510982] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
174  [   38.512209] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
175  [   39.711132] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
176  [   39.711210] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
177  [   39.711779] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
178  [   39.712735] livepatch: 'livepatch_callbacks_demo': unpatching complete
179
180  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
181  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
182  [   42.534183] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
183
184
185Test 2
186------
187
188This test is similar to the previous test, but (un)load the livepatch
189module before the target kernel module.  This tests the livepatch core's
190module_coming handler:
191
192- load livepatch
193- load target module
194- disable livepatch
195- unload livepatch
196- unload target module
197
198
199On livepatch enable, only pre/post-patch callbacks are executed for
200currently loaded klp_objects, in this case, vmlinux:
201
202  % insmod samples/livepatch/livepatch-callbacks-demo.ko
203  [   44.553328] livepatch: enabling patch 'livepatch_callbacks_demo'
204  [   44.553997] livepatch: 'livepatch_callbacks_demo': initializing patching transition
205  [   44.554049] livepatch_callbacks_demo: pre_patch_callback: vmlinux
206  [   44.554845] livepatch: 'livepatch_callbacks_demo': starting patching transition
207  [   45.727128] livepatch: 'livepatch_callbacks_demo': completing patching transition
208  [   45.727212] livepatch_callbacks_demo: post_patch_callback: vmlinux
209  [   45.727961] livepatch: 'livepatch_callbacks_demo': patching complete
210
211When a targeted module is subsequently loaded, only its pre/post-patch
212callbacks are executed:
213
214  % insmod samples/livepatch/livepatch-callbacks-mod.ko
215  [   46.560845] livepatch: applying patch 'livepatch_callbacks_demo' to loading module 'livepatch_callbacks_mod'
216  [   46.561988] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
217  [   46.563452] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
218  [   46.565495] livepatch_callbacks_mod: livepatch_callbacks_mod_init
219
220On livepatch disable, all currently loaded klp_objects' (vmlinux and
221livepatch_callbacks_mod) pre/post-unpatch callbacks are executed:
222
223  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
224  [   48.568885] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
225  [   48.568910] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
226  [   48.569441] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
227  [   48.570502] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
228  [   49.759091] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
229  [   49.759171] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
230  [   49.759742] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
231  [   49.760690] livepatch: 'livepatch_callbacks_demo': unpatching complete
232
233  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
234  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
235  [   52.592283] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
236
237
238Test 3
239------
240
241Test loading the livepatch after a targeted kernel module, then unload
242the kernel module before disabling the livepatch.  This tests the
243livepatch core's module_going handler:
244
245- load target module
246- load livepatch
247- unload target module
248- disable livepatch
249- unload livepatch
250
251First load a target module, then the livepatch:
252
253  % insmod samples/livepatch/livepatch-callbacks-mod.ko
254  [   54.607948] livepatch_callbacks_mod: livepatch_callbacks_mod_init
255
256  % insmod samples/livepatch/livepatch-callbacks-demo.ko
257  [   56.613919] livepatch: enabling patch 'livepatch_callbacks_demo'
258  [   56.614411] livepatch: 'livepatch_callbacks_demo': initializing patching transition
259  [   56.614436] livepatch_callbacks_demo: pre_patch_callback: vmlinux
260  [   56.614818] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
261  [   56.615656] livepatch: 'livepatch_callbacks_demo': starting patching transition
262  [   57.759070] livepatch: 'livepatch_callbacks_demo': completing patching transition
263  [   57.759147] livepatch_callbacks_demo: post_patch_callback: vmlinux
264  [   57.759621] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_LIVE] Normal state
265  [   57.760307] livepatch: 'livepatch_callbacks_demo': patching complete
266
267When a target module is unloaded, the livepatch is only reverted from
268that klp_object (livepatch_callbacks_mod).  As such, only its pre and
269post-unpatch callbacks are executed when this occurs:
270
271  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
272  [   58.623409] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
273  [   58.623903] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
274  [   58.624658] livepatch: reverting patch 'livepatch_callbacks_demo' on unloading module 'livepatch_callbacks_mod'
275  [   58.625305] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
276
277When the livepatch is disabled, pre and post-unpatch callbacks are run
278for the remaining klp_object, vmlinux:
279
280  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
281  [   60.638420] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
282  [   60.638444] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
283  [   60.638996] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
284  [   61.727088] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
285  [   61.727165] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
286  [   61.727985] livepatch: 'livepatch_callbacks_demo': unpatching complete
287
288  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
289
290
291Test 4
292------
293
294This test is similar to the previous test, however the livepatch is
295loaded first.  This tests the livepatch core's module_coming and
296module_going handlers:
297
298- load livepatch
299- load target module
300- unload target module
301- disable livepatch
302- unload livepatch
303
304First load the livepatch:
305
306  % insmod samples/livepatch/livepatch-callbacks-demo.ko
307  [   64.661552] livepatch: enabling patch 'livepatch_callbacks_demo'
308  [   64.662147] livepatch: 'livepatch_callbacks_demo': initializing patching transition
309  [   64.662175] livepatch_callbacks_demo: pre_patch_callback: vmlinux
310  [   64.662850] livepatch: 'livepatch_callbacks_demo': starting patching transition
311  [   65.695056] livepatch: 'livepatch_callbacks_demo': completing patching transition
312  [   65.695147] livepatch_callbacks_demo: post_patch_callback: vmlinux
313  [   65.695561] livepatch: 'livepatch_callbacks_demo': patching complete
314
315When a targeted kernel module is subsequently loaded, only its
316pre/post-patch callbacks are executed:
317
318  % insmod samples/livepatch/livepatch-callbacks-mod.ko
319  [   66.669196] livepatch: applying patch 'livepatch_callbacks_demo' to loading module 'livepatch_callbacks_mod'
320  [   66.669882] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
321  [   66.670744] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
322  [   66.672873] livepatch_callbacks_mod: livepatch_callbacks_mod_init
323
324When the target module is unloaded, the livepatch is only reverted from
325the livepatch_callbacks_mod klp_object.  As such, only pre and
326post-unpatch callbacks are executed when this occurs:
327
328  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
329  [   68.680065] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
330  [   68.680688] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
331  [   68.681452] livepatch: reverting patch 'livepatch_callbacks_demo' on unloading module 'livepatch_callbacks_mod'
332  [   68.682094] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
333
334  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
335  [   70.689225] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
336  [   70.689256] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
337  [   70.689882] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
338  [   71.711080] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
339  [   71.711481] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
340  [   71.711988] livepatch: 'livepatch_callbacks_demo': unpatching complete
341
342  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
343
344
345Test 5
346------
347
348A simple test of loading a livepatch without one of its patch target
349klp_objects ever loaded (livepatch_callbacks_mod):
350
351- load livepatch
352- disable livepatch
353- unload livepatch
354
355Load the livepatch:
356
357  % insmod samples/livepatch/livepatch-callbacks-demo.ko
358  [   74.711081] livepatch: enabling patch 'livepatch_callbacks_demo'
359  [   74.711595] livepatch: 'livepatch_callbacks_demo': initializing patching transition
360  [   74.711639] livepatch_callbacks_demo: pre_patch_callback: vmlinux
361  [   74.712272] livepatch: 'livepatch_callbacks_demo': starting patching transition
362  [   75.743137] livepatch: 'livepatch_callbacks_demo': completing patching transition
363  [   75.743219] livepatch_callbacks_demo: post_patch_callback: vmlinux
364  [   75.743867] livepatch: 'livepatch_callbacks_demo': patching complete
365
366As expected, only pre/post-(un)patch handlers are executed for vmlinux:
367
368  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
369  [   76.716254] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
370  [   76.716278] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
371  [   76.716666] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
372  [   77.727089] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
373  [   77.727194] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
374  [   77.727907] livepatch: 'livepatch_callbacks_demo': unpatching complete
375
376  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
377
378
379Test 6
380------
381
382Test a scenario where a vmlinux pre-patch callback returns a non-zero
383status (ie, failure):
384
385- load target module
386- load livepatch -ENODEV
387- unload target module
388
389First load a target module:
390
391  % insmod samples/livepatch/livepatch-callbacks-mod.ko
392  [   80.740520] livepatch_callbacks_mod: livepatch_callbacks_mod_init
393
394Load the livepatch module, setting its 'pre_patch_ret' value to -19
395(-ENODEV).  When its vmlinux pre-patch callback executed, this status
396code will propagate back to the module-loading subsystem.  The result is
397that the insmod command refuses to load the livepatch module:
398
399  % insmod samples/livepatch/livepatch-callbacks-demo.ko pre_patch_ret=-19
400  [   82.747326] livepatch: enabling patch 'livepatch_callbacks_demo'
401  [   82.747743] livepatch: 'livepatch_callbacks_demo': initializing patching transition
402  [   82.747767] livepatch_callbacks_demo: pre_patch_callback: vmlinux
403  [   82.748237] livepatch: pre-patch callback failed for object 'vmlinux'
404  [   82.748637] livepatch: failed to enable patch 'livepatch_callbacks_demo'
405  [   82.749059] livepatch: 'livepatch_callbacks_demo': canceling transition, going to unpatch
406  [   82.749060] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
407  [   82.749868] livepatch: 'livepatch_callbacks_demo': unpatching complete
408  [   82.765809] insmod: ERROR: could not insert module samples/livepatch/livepatch-callbacks-demo.ko: No such device
409
410  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
411  [   84.774238] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
412
413
414Test 7
415------
416
417Similar to the previous test, setup a livepatch such that its vmlinux
418pre-patch callback returns success.  However, when a targeted kernel
419module is later loaded, have the livepatch return a failing status code:
420
421- load livepatch
422- setup -ENODEV
423- load target module
424- disable livepatch
425- unload livepatch
426
427Load the livepatch, notice vmlinux pre-patch callback succeeds:
428
429  % insmod samples/livepatch/livepatch-callbacks-demo.ko
430  [   86.787845] livepatch: enabling patch 'livepatch_callbacks_demo'
431  [   86.788325] livepatch: 'livepatch_callbacks_demo': initializing patching transition
432  [   86.788427] livepatch_callbacks_demo: pre_patch_callback: vmlinux
433  [   86.788821] livepatch: 'livepatch_callbacks_demo': starting patching transition
434  [   87.711069] livepatch: 'livepatch_callbacks_demo': completing patching transition
435  [   87.711143] livepatch_callbacks_demo: post_patch_callback: vmlinux
436  [   87.711886] livepatch: 'livepatch_callbacks_demo': patching complete
437
438Set a trap so subsequent pre-patch callbacks to this livepatch will
439return -ENODEV:
440
441  % echo -19 > /sys/module/livepatch_callbacks_demo/parameters/pre_patch_ret
442
443The livepatch pre-patch callback for subsequently loaded target modules
444will return failure, so the module loader refuses to load the kernel
445module.  Notice that no post-patch or pre/post-unpatch callbacks are
446executed for this klp_object:
447
448  % insmod samples/livepatch/livepatch-callbacks-mod.ko
449  [   90.796976] livepatch: applying patch 'livepatch_callbacks_demo' to loading module 'livepatch_callbacks_mod'
450  [   90.797834] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
451  [   90.798900] livepatch: pre-patch callback failed for object 'livepatch_callbacks_mod'
452  [   90.799652] livepatch: patch 'livepatch_callbacks_demo' failed for module 'livepatch_callbacks_mod', refusing to load module 'livepatch_callbacks_mod'
453  [   90.819737] insmod: ERROR: could not insert module samples/livepatch/livepatch-callbacks-mod.ko: No such device
454
455However, pre/post-unpatch callbacks run for the vmlinux klp_object:
456
457  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
458  [   92.823547] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
459  [   92.823573] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
460  [   92.824331] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
461  [   93.727128] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
462  [   93.727327] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
463  [   93.727861] livepatch: 'livepatch_callbacks_demo': unpatching complete
464
465  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
466
467
468Test 8
469------
470
471Test loading multiple targeted kernel modules.  This test-case is
472mainly for comparing with the next test-case.
473
474- load busy target module (0s sleep),
475- load livepatch
476- load target module
477- unload target module
478- disable livepatch
479- unload livepatch
480- unload busy target module
481
482
483Load a target "busy" kernel module which kicks off a worker function
484that immediately exits:
485
486  % insmod samples/livepatch/livepatch-callbacks-busymod.ko sleep_secs=0
487  [   96.910107] livepatch_callbacks_busymod: livepatch_callbacks_mod_init
488  [   96.910600] livepatch_callbacks_busymod: busymod_work_func, sleeping 0 seconds ...
489  [   96.913024] livepatch_callbacks_busymod: busymod_work_func exit
490
491Proceed with loading the livepatch and another ordinary target module,
492notice that the post-patch callbacks are executed and the transition
493completes quickly:
494
495  % insmod samples/livepatch/livepatch-callbacks-demo.ko
496  [   98.917892] livepatch: enabling patch 'livepatch_callbacks_demo'
497  [   98.918426] livepatch: 'livepatch_callbacks_demo': initializing patching transition
498  [   98.918453] livepatch_callbacks_demo: pre_patch_callback: vmlinux
499  [   98.918955] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
500  [   98.923835] livepatch: 'livepatch_callbacks_demo': starting patching transition
501  [   99.743104] livepatch: 'livepatch_callbacks_demo': completing patching transition
502  [   99.743156] livepatch_callbacks_demo: post_patch_callback: vmlinux
503  [   99.743679] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
504  [   99.744616] livepatch: 'livepatch_callbacks_demo': patching complete
505
506  % insmod samples/livepatch/livepatch-callbacks-mod.ko
507  [  100.930955] livepatch: applying patch 'livepatch_callbacks_demo' to loading module 'livepatch_callbacks_mod'
508  [  100.931668] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
509  [  100.932645] livepatch_callbacks_demo: post_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
510  [  100.934125] livepatch_callbacks_mod: livepatch_callbacks_mod_init
511
512  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
513  [  102.942805] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
514  [  102.943640] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
515  [  102.944585] livepatch: reverting patch 'livepatch_callbacks_demo' on unloading module 'livepatch_callbacks_mod'
516  [  102.945455] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
517
518  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
519  [  104.953815] livepatch: 'livepatch_callbacks_demo': initializing unpatching transition
520  [  104.953838] livepatch_callbacks_demo: pre_unpatch_callback: vmlinux
521  [  104.954431] livepatch_callbacks_demo: pre_unpatch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
522  [  104.955426] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
523  [  106.719073] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
524  [  106.722633] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
525  [  106.723282] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
526  [  106.724279] livepatch: 'livepatch_callbacks_demo': unpatching complete
527
528  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
529  % rmmod samples/livepatch/livepatch-callbacks-busymod.ko
530  [  108.975660] livepatch_callbacks_busymod: livepatch_callbacks_mod_exit
531
532
533Test 9
534------
535
536A similar test as the previous one, but force the "busy" kernel module
537to do longer work.
538
539The livepatching core will refuse to patch a task that is currently
540executing a to-be-patched function -- the consistency model stalls the
541current patch transition until this safety-check is met.  Test a
542scenario where one of a livepatch's target klp_objects sits on such a
543function for a long time.  Meanwhile, load and unload other target
544kernel modules while the livepatch transition is in progress.
545
546- load busy target module (30s sleep)
547- load livepatch
548- load target module
549- unload target module
550- disable livepatch
551- unload livepatch
552- unload busy target module
553
554
555Load the "busy" kernel module, this time make it do 30 seconds worth of
556work:
557
558  % insmod samples/livepatch/livepatch-callbacks-busymod.ko sleep_secs=30
559  [  110.993362] livepatch_callbacks_busymod: livepatch_callbacks_mod_init
560  [  110.994059] livepatch_callbacks_busymod: busymod_work_func, sleeping 30 seconds ...
561
562Meanwhile, the livepatch is loaded.  Notice that the patch transition
563does not complete as the targeted "busy" module is sitting on a
564to-be-patched function:
565
566  % insmod samples/livepatch/livepatch-callbacks-demo.ko
567  [  113.000309] livepatch: enabling patch 'livepatch_callbacks_demo'
568  [  113.000764] livepatch: 'livepatch_callbacks_demo': initializing patching transition
569  [  113.000791] livepatch_callbacks_demo: pre_patch_callback: vmlinux
570  [  113.001289] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
571  [  113.005208] livepatch: 'livepatch_callbacks_demo': starting patching transition
572
573Load a second target module (this one is an ordinary idle kernel
574module).  Note that *no* post-patch callbacks will be executed while the
575livepatch is still in transition:
576
577  % insmod samples/livepatch/livepatch-callbacks-mod.ko
578  [  115.012740] livepatch: applying patch 'livepatch_callbacks_demo' to loading module 'livepatch_callbacks_mod'
579  [  115.013406] livepatch_callbacks_demo: pre_patch_callback: livepatch_callbacks_mod -> [MODULE_STATE_COMING] Full formed, running module_init
580  [  115.015315] livepatch_callbacks_mod: livepatch_callbacks_mod_init
581
582Request an unload of the simple kernel module.  The patch is still
583transitioning, so its pre-unpatch callbacks are skipped:
584
585  % rmmod samples/livepatch/livepatch-callbacks-mod.ko
586  [  117.022626] livepatch_callbacks_mod: livepatch_callbacks_mod_exit
587  [  117.023376] livepatch: reverting patch 'livepatch_callbacks_demo' on unloading module 'livepatch_callbacks_mod'
588  [  117.024533] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_mod -> [MODULE_STATE_GOING] Going away
589
590Finally the livepatch is disabled.  Since none of the patch's
591klp_object's post-patch callbacks executed, the remaining klp_object's
592pre-unpatch callbacks are skipped:
593
594  % echo 0 > /sys/kernel/livepatch/livepatch_callbacks_demo/enabled
595  [  119.035408] livepatch: 'livepatch_callbacks_demo': reversing transition from patching to unpatching
596  [  119.035485] livepatch: 'livepatch_callbacks_demo': starting unpatching transition
597  [  119.711166] livepatch: 'livepatch_callbacks_demo': completing unpatching transition
598  [  119.714179] livepatch_callbacks_demo: post_unpatch_callback: vmlinux
599  [  119.714653] livepatch_callbacks_demo: post_unpatch_callback: livepatch_callbacks_busymod -> [MODULE_STATE_LIVE] Normal state
600  [  119.715437] livepatch: 'livepatch_callbacks_demo': unpatching complete
601
602  % rmmod samples/livepatch/livepatch-callbacks-demo.ko
603  % rmmod samples/livepatch/livepatch-callbacks-busymod.ko
604  [  141.279111] livepatch_callbacks_busymod: busymod_work_func exit
605  [  141.279760] livepatch_callbacks_busymod: livepatch_callbacks_mod_exit
606