Blame view

kernel/livepatch/state.c 3.07 KB
73727f4da   Petr Mladek   livepatch: Basic ...
1
2
3
4
5
6
7
8
9
10
11
  // SPDX-License-Identifier: GPL-2.0-or-later
  /*
   * system_state.c - State of the system modified by livepatches
   *
   * Copyright (C) 2019 SUSE
   */
  
  #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  
  #include <linux/livepatch.h>
  #include "core.h"
92c9abf5e   Petr Mladek   livepatch: Allow ...
12
  #include "state.h"
73727f4da   Petr Mladek   livepatch: Basic ...
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
  #include "transition.h"
  
  #define klp_for_each_state(patch, state)		\
  	for (state = patch->states; state && state->id; state++)
  
  /**
   * klp_get_state() - get information about system state modified by
   *	the given patch
   * @patch:	livepatch that modifies the given system state
   * @id:		custom identifier of the modified system state
   *
   * Checks whether the given patch modifies the given system state.
   *
   * The function can be called either from pre/post (un)patch
   * callbacks or from the kernel code added by the livepatch.
   *
   * Return: pointer to struct klp_state when found, otherwise NULL.
   */
  struct klp_state *klp_get_state(struct klp_patch *patch, unsigned long id)
  {
  	struct klp_state *state;
  
  	klp_for_each_state(patch, state) {
  		if (state->id == id)
  			return state;
  	}
  
  	return NULL;
  }
  EXPORT_SYMBOL_GPL(klp_get_state);
  
  /**
   * klp_get_prev_state() - get information about system state modified by
   *	the already installed livepatches
   * @id:		custom identifier of the modified system state
   *
   * Checks whether already installed livepatches modify the given
   * system state.
   *
   * The same system state can be modified by more non-cumulative
   * livepatches. It is expected that the latest livepatch has
   * the most up-to-date information.
   *
   * The function can be called only during transition when a new
   * livepatch is being enabled or when such a transition is reverted.
7b7b8a2c9   Randy Dunlap   kernel/: fix repe...
58
   * It is typically called only from pre/post (un)patch
73727f4da   Petr Mladek   livepatch: Basic ...
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
   * callbacks.
   *
   * Return: pointer to the latest struct klp_state from already
   *	installed livepatches, NULL when not found.
   */
  struct klp_state *klp_get_prev_state(unsigned long id)
  {
  	struct klp_patch *patch;
  	struct klp_state *state, *last_state = NULL;
  
  	if (WARN_ON_ONCE(!klp_transition_patch))
  		return NULL;
  
  	klp_for_each_patch(patch) {
  		if (patch == klp_transition_patch)
  			goto out;
  
  		state = klp_get_state(patch, id);
  		if (state)
  			last_state = state;
  	}
  
  out:
  	return last_state;
  }
  EXPORT_SYMBOL_GPL(klp_get_prev_state);
92c9abf5e   Petr Mladek   livepatch: Allow ...
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
  
  /* Check if the patch is able to deal with the existing system state. */
  static bool klp_is_state_compatible(struct klp_patch *patch,
  				    struct klp_state *old_state)
  {
  	struct klp_state *state;
  
  	state = klp_get_state(patch, old_state->id);
  
  	/* A cumulative livepatch must handle all already modified states. */
  	if (!state)
  		return !patch->replace;
  
  	return state->version >= old_state->version;
  }
  
  /*
   * Check that the new livepatch will not break the existing system states.
   * Cumulative patches must handle all already modified states.
   * Non-cumulative patches can touch already modified states.
   */
  bool klp_is_patch_compatible(struct klp_patch *patch)
  {
  	struct klp_patch *old_patch;
  	struct klp_state *old_state;
  
  	klp_for_each_patch(old_patch) {
  		klp_for_each_state(old_patch, old_state) {
  			if (!klp_is_state_compatible(patch, old_state))
  				return false;
  		}
  	}
  
  	return true;
  }