hc
2024-08-13 f258bb3ae540ccc311fd344a0121bba1928b85dd
1
2
3
4
5
6
7
8
9
10
11
12
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
58
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
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
120
121
122
123
124
125
126
127
128
129
// SPDX-License-Identifier: GPL-2.0
// Copyright (C) 2017 Arm Ltd.
#ifndef __LINUX_ARM_SDEI_H
#define __LINUX_ARM_SDEI_H
 
#include <uapi/linux/arm_sdei.h>
 
#include <acpi/ghes.h>
 
#ifdef CONFIG_ARM_SDE_INTERFACE
#include <asm/sdei.h>
#endif
 
/* Arch code should override this to set the entry point from firmware... */
#ifndef sdei_arch_get_entry_point
#define sdei_arch_get_entry_point(conduit)    (0)
#endif
 
/*
 * When an event occurs sdei_event_handler() will call a user-provided callback
 * like this in NMI context on the CPU that received the event.
 */
typedef int (sdei_event_callback)(u32 event, struct pt_regs *regs, void *arg);
 
/*
 * Register your callback to claim an event. The event must be described
 * by firmware.
 */
int sdei_event_register(u32 event_num, sdei_event_callback *cb, void *arg);
 
/*
 * Calls to sdei_event_unregister() may return EINPROGRESS. Keep calling
 * it until it succeeds.
 */
int sdei_event_unregister(u32 event_num);
 
int sdei_event_enable(u32 event_num);
int sdei_event_disable(u32 event_num);
 
#ifdef CONFIG_FIQ_DEBUGGER_TRUST_ZONE
#ifdef CONFIG_ARM_SDE_INTERFACE
int sdei_event_enable_nolock(u32 event_num);
int sdei_event_disable_nolock(u32 event_num);
int sdei_event_routing_set_nolock(u32 event_num, unsigned long flags,
                 unsigned long affinity);
int sdei_event_routing_set(u32 event_num, unsigned long flags,
              unsigned long affinity);
int sdei_interrupt_bind(u32 intr_num, u32 *event_num);
int sdei_interrupt_release(u32 event_num);
#else
static inline int sdei_event_enable_nolock(u32 event_num)
{
   return SDEI_NOT_SUPPORTED;
}
 
static inline int sdei_event_disable_nolock(u32 event_num)
{
   return SDEI_NOT_SUPPORTED;
}
 
static inline int sdei_event_routing_set_nolock(u32 event_num,
                       unsigned long flags,
                       unsigned long affinity)
{
   return SDEI_NOT_SUPPORTED;
}
 
static inline int sdei_event_routing_set(u32 event_num,
                    unsigned long flags,
                    unsigned long affinity)
{
   return SDEI_NOT_SUPPORTED;
}
 
static inline int sdei_interrupt_bind(u32 intr_num, u32 *event_num)
{
   return SDEI_NOT_SUPPORTED;
}
 
static inline int sdei_interrupt_release(u32 event_num)
{
   return SDEI_NOT_SUPPORTED;
}
#endif /* CONFIG_ARM_SDE_INTERFACE */
#endif /* CONFIG_FIQ_DEBUGGER_TRUST_ZONE */
 
/* GHES register/unregister helpers */
int sdei_register_ghes(struct ghes *ghes, sdei_event_callback *normal_cb,
              sdei_event_callback *critical_cb);
int sdei_unregister_ghes(struct ghes *ghes);
 
#ifdef CONFIG_ARM_SDE_INTERFACE
/* For use by arch code when CPU hotplug notifiers are not appropriate. */
int sdei_mask_local_cpu(void);
int sdei_unmask_local_cpu(void);
#else
static inline int sdei_mask_local_cpu(void) { return 0; }
static inline int sdei_unmask_local_cpu(void) { return 0; }
#endif /* CONFIG_ARM_SDE_INTERFACE */
 
 
/*
 * This struct represents an event that has been registered. The driver
 * maintains a list of all events, and which ones are registered. (Private
 * events have one entry in the list, but are registered on each CPU).
 * A pointer to this struct is passed to firmware, and back to the event
 * handler. The event handler can then use this to invoke the registered
 * callback, without having to walk the list.
 *
 * For CPU private events, this structure is per-cpu.
 */
struct sdei_registered_event {
   /* For use by arch code: */
   struct pt_regs          interrupted_regs;
 
   sdei_event_callback    *callback;
   void            *callback_arg;
   u32             event_num;
   u8             priority;
};
 
/* The arch code entry point should then call this when an event arrives. */
int notrace sdei_event_handler(struct pt_regs *regs,
                  struct sdei_registered_event *arg);
 
/* arch code may use this to retrieve the extra registers. */
int sdei_api_event_context(u32 query, u64 *result);
 
#endif /* __LINUX_ARM_SDEI_H */