pacemaker  2.1.7-0f7f88312f
Scalable High-Availability cluster resource manager
internal.h
Go to the documentation of this file.
1 /*
2  * Copyright 2015-2023 the Pacemaker project contributors
3  *
4  * The version control history for this file may have further details.
5  *
6  * This source code is licensed under the GNU Lesser General Public License
7  * version 2.1 or later (LGPLv2.1+) WITHOUT ANY WARRANTY.
8  */
9 
10 #ifndef CRM_COMMON_INTERNAL__H
11 #define CRM_COMMON_INTERNAL__H
12 
13 #include <unistd.h> // pid_t, getpid()
14 #include <stdbool.h> // bool
15 #include <stdint.h> // uint8_t, uint64_t
16 
17 #include <glib.h> // guint, GList, GHashTable
18 #include <libxml/tree.h> // xmlNode
19 
20 #include <crm/common/util.h> // crm_strdup_printf()
21 #include <crm/common/logging.h> // do_crm_log_unlikely(), etc.
22 #include <crm/common/mainloop.h> // mainloop_io_t, struct ipc_client_callbacks
26 #include <crm/common/io_internal.h>
32 
33 /* This says whether the current application is a Pacemaker daemon or not,
34  * and is used to change default logging settings such as whether to log to
35  * stderr, etc., as well as a few other details such as whether blackbox signal
36  * handling is enabled.
37  *
38  * It is set when logging is initialized, and does not need to be set directly.
39  */
40 extern bool pcmk__is_daemon;
41 
43 extern char *pcmk__our_nodename;
44 
45 // Number of elements in a statically defined array
46 #define PCMK__NELEM(a) ((int) (sizeof(a)/sizeof(a[0])) )
47 
48 #if SUPPORT_CIBSECRETS
49 /* internal CIB utilities (from cib_secrets.c) */
50 
51 int pcmk__substitute_secrets(const char *rsc_id, GHashTable *params);
52 #endif
53 
54 
55 /* internal main loop utilities (from mainloop.c) */
56 
57 int pcmk__add_mainloop_ipc(crm_ipc_t *ipc, int priority, void *userdata,
58  const struct ipc_client_callbacks *callbacks,
59  mainloop_io_t **source);
61 
62 
63 /* internal node-related XML utilities (from nodes.c) */
64 
73 void pcmk__xe_add_node(xmlNode *xml, const char *node, int nodeid);
74 
75 
76 /* internal name/value utilities (from nvpair.c) */
77 
78 int pcmk__scan_nvpair(const char *input, char **name, char **value);
79 char *pcmk__format_nvpair(const char *name, const char *value,
80  const char *units);
81 
90 void
91 pcmk__xe_set_bool_attr(xmlNodePtr node, const char *name, bool value);
92 
103 bool
104 pcmk__xe_attr_is_true(const xmlNode *node, const char *name);
105 
121 int
122 pcmk__xe_get_bool_attr(const xmlNode *node, const char *name, bool *value);
123 
124 
125 /* internal procfs utilities (from procfs.c) */
126 
127 pid_t pcmk__procfs_pid_of(const char *name);
128 unsigned int pcmk__procfs_num_cores(void);
129 int pcmk__procfs_pid2path(pid_t pid, char path[], size_t path_size);
130 bool pcmk__procfs_has_pids(void);
131 
132 /* internal XML schema functions (from xml.c) */
133 
134 void crm_schema_init(void);
135 void crm_schema_cleanup(void);
136 
137 
138 /* internal functions related to process IDs (from pid.c) */
139 
156 int pcmk__pid_active(pid_t pid, const char *daemon);
157 
158 int pcmk__read_pidfile(const char *filename, pid_t *pid);
159 int pcmk__pidfile_matches(const char *filename, pid_t expected_pid,
160  const char *expected_name, pid_t *pid);
161 int pcmk__lock_pidfile(const char *filename, const char *name);
162 
163 
164 // bitwise arithmetic utilities
165 
181 static inline uint64_t
182 pcmk__set_flags_as(const char *function, int line, uint8_t log_level,
183  const char *flag_type, const char *target,
184  uint64_t flag_group, uint64_t flags, const char *flags_str)
185 {
186  uint64_t result = flag_group | flags;
187 
188  if (result != flag_group) {
189  do_crm_log_unlikely(log_level,
190  "%s flags %#.8llx (%s) for %s set by %s:%d",
191  ((flag_type == NULL)? "Group of" : flag_type),
192  (unsigned long long) flags,
193  ((flags_str == NULL)? "flags" : flags_str),
194  ((target == NULL)? "target" : target),
195  function, line);
196  }
197  return result;
198 }
199 
215 static inline uint64_t
216 pcmk__clear_flags_as(const char *function, int line, uint8_t log_level,
217  const char *flag_type, const char *target,
218  uint64_t flag_group, uint64_t flags, const char *flags_str)
219 {
220  uint64_t result = flag_group & ~flags;
221 
222  if (result != flag_group) {
223  do_crm_log_unlikely(log_level,
224  "%s flags %#.8llx (%s) for %s cleared by %s:%d",
225  ((flag_type == NULL)? "Group of" : flag_type),
226  (unsigned long long) flags,
227  ((flags_str == NULL)? "flags" : flags_str),
228  ((target == NULL)? "target" : target),
229  function, line);
230  }
231  return result;
232 }
233 
234 // miscellaneous utilities (from utils.c)
235 
236 void pcmk__daemonize(const char *name, const char *pidfile);
237 void pcmk__panic(const char *origin);
238 pid_t pcmk__locate_sbd(void);
239 void pcmk__sleep_ms(unsigned int ms);
240 
241 extern int pcmk__score_red;
242 extern int pcmk__score_green;
243 extern int pcmk__score_yellow;
244 
257 static inline void *
258 pcmk__realloc(void *ptr, size_t size)
259 {
260  void *new_ptr;
261 
262  // realloc(p, 0) can replace free(p) but this wrapper can't
263  CRM_ASSERT(size > 0);
264 
265  new_ptr = realloc(ptr, size);
266  if (new_ptr == NULL) {
267  free(ptr);
268  abort();
269  }
270  return new_ptr;
271 }
272 
273 
274 static inline char *
275 pcmk__getpid_s(void)
276 {
277  return crm_strdup_printf("%lu", (unsigned long) getpid());
278 }
279 
280 // More efficient than g_list_length(list) == 1
281 static inline bool
282 pcmk__list_of_1(GList *list)
283 {
284  return list && (list->next == NULL);
285 }
286 
287 // More efficient than g_list_length(list) > 1
288 static inline bool
289 pcmk__list_of_multiple(GList *list)
290 {
291  return list && (list->next != NULL);
292 }
293 
294 /* convenience functions for failure-related node attributes */
295 
296 #define PCMK__FAIL_COUNT_PREFIX "fail-count"
297 #define PCMK__LAST_FAILURE_PREFIX "last-failure"
298 
316 static inline char *
317 pcmk__fail_attr_name(const char *prefix, const char *rsc_id, const char *op,
318  guint interval_ms)
319 {
320  CRM_CHECK(prefix && rsc_id && op, return NULL);
321  return crm_strdup_printf("%s-%s#%s_%u", prefix, rsc_id, op, interval_ms);
322 }
323 
324 static inline char *
325 pcmk__failcount_name(const char *rsc_id, const char *op, guint interval_ms)
326 {
327  return pcmk__fail_attr_name(PCMK__FAIL_COUNT_PREFIX, rsc_id, op,
328  interval_ms);
329 }
330 
331 static inline char *
332 pcmk__lastfailure_name(const char *rsc_id, const char *op, guint interval_ms)
333 {
334  return pcmk__fail_attr_name(PCMK__LAST_FAILURE_PREFIX, rsc_id, op,
335  interval_ms);
336 }
337 
338 // internal resource agent functions (from agents.c)
339 int pcmk__effective_rc(int rc);
340 
341 #endif /* CRM_COMMON_INTERNAL__H */
#define CRM_CHECK(expr, failure_action)
Definition: logging.h:238
uint32_t size
Definition: cpg.c:49
void crm_schema_init(void)
Definition: schemas.c:377
int pcmk__read_pidfile(const char *filename, pid_t *pid)
Definition: pid.c:111
const char * name
Definition: cib.c:26
int pcmk__substitute_secrets(const char *rsc_id, GHashTable *params)
Definition: cib_secrets.c:96
struct mainloop_timer_s mainloop_timer_t
Definition: mainloop.h:35
struct mainloop_io_s mainloop_io_t
Definition: mainloop.h:33
void crm_schema_cleanup(void)
Definition: schemas.c:523
void pcmk__xe_set_bool_attr(xmlNodePtr node, const char *name, bool value)
Definition: nvpair.c:872
bool pcmk__xe_attr_is_true(const xmlNode *node, const char *name)
Definition: nvpair.c:905
Wrappers for and extensions to glib mainloop.
#define PCMK__LAST_FAILURE_PREFIX
Definition: internal.h:297
Wrappers for and extensions to libqb logging.
bool pcmk__procfs_has_pids(void)
Definition: procfs.c:211
int pcmk__effective_rc(int rc)
Definition: agents.c:71
char * pcmk__format_nvpair(const char *name, const char *value, const char *units)
Definition: nvpair.c:284
#define do_crm_log_unlikely(level, fmt, args...)
Log a message that is likely to be filtered out.
Definition: logging.h:200
void pcmk__sleep_ms(unsigned int ms)
Definition: utils.c:561
uint32_t pid
Definition: cpg.c:46
struct crm_ipc_s crm_ipc_t
Definition: ipc.h:165
Utility functions.
unsigned int pcmk__procfs_num_cores(void)
Definition: procfs.c:145
char * crm_strdup_printf(char const *format,...) G_GNUC_PRINTF(1
int pcmk__scan_nvpair(const char *input, char **name, char **value)
Definition: nvpair.c:221
int pcmk__xe_get_bool_attr(const xmlNode *node, const char *name, bool *value)
Definition: nvpair.c:878
bool pcmk__is_daemon
Definition: logging.c:47
#define PCMK__FAIL_COUNT_PREFIX
Definition: internal.h:296
int pcmk__pid_active(pid_t pid, const char *daemon)
Definition: pid.c:23
pid_t pcmk__locate_sbd(void)
Definition: watchdog.c:194
int pcmk__add_mainloop_ipc(crm_ipc_t *ipc, int priority, void *userdata, const struct ipc_client_callbacks *callbacks, mainloop_io_t **source)
Connect to IPC and add it as a main loop source.
Definition: mainloop.c:864
pid_t pcmk__procfs_pid_of(const char *name)
Definition: procfs.c:111
const char * target
Definition: pcmk_fence.c:29
int pcmk__score_green
Definition: scores.c:21
void pcmk__daemonize(const char *name, const char *pidfile)
Definition: utils.c:421
int pcmk__lock_pidfile(const char *filename, const char *name)
Definition: pid.c:214
pcmk__action_result_t result
Definition: pcmk_fence.c:35
int pcmk__score_red
Definition: scores.c:20
const char * path
Definition: cib.c:28
char * pcmk__our_nodename
Node name of the local node.
Definition: logging.c:48
#define CRM_ASSERT(expr)
Definition: results.h:42
xmlNode * input
int pcmk__procfs_pid2path(pid_t pid, char path[], size_t path_size)
Definition: procfs.c:179
int pcmk__score_yellow
Definition: scores.c:22
int pcmk__pidfile_matches(const char *filename, pid_t expected_pid, const char *expected_name, pid_t *pid)
Definition: pid.c:172
void pcmk__xe_add_node(xmlNode *xml, const char *node, int nodeid)
Definition: nodes.c:15
void pcmk__panic(const char *origin)
Definition: watchdog.c:161
guint pcmk__mainloop_timer_get_period(const mainloop_timer_t *timer)
Get period for mainloop timer.
Definition: mainloop.c:910
uint64_t flags
Definition: remote.c:215