Skip to content

Commit 356bf97

Browse files
armandomontanezCQ Bot Account
authored and
CQ Bot Account
committed
pw_cpu_exception: Split facade
The pw_cpu_exception facade is comprised of three parts: the part implemented by the architecture, and the part implemented by the application, and some supporting libraries that make it easier to dump CPU state. Since these are provided independently, the facade has been split into three facades: 1. entry 2. handler 3. support Change-Id: I48c92f42b208f79596740f1bec59ed3d29e277a8 Reviewed-on: https://pigweed-review.googlesource.com/c/pigweed/pigweed/+/11646 Commit-Queue: Armando Montanez <amontanez@google.com> Reviewed-by: Wyatt Hepler <hepler@google.com>
1 parent ce87bc0 commit 356bf97

File tree

19 files changed

+387
-164
lines changed

19 files changed

+387
-164
lines changed

BUILD.gn

+1-1
Original file line numberDiff line numberDiff line change
@@ -185,7 +185,7 @@ if (current_toolchain != default_toolchain) {
185185
import("$dir_pw_cpu_exception/backend.gni")
186186

187187
# TODO(pwbug/17): Re-think when Pigweed config system is added.
188-
if (pw_cpu_exception_BACKEND == dir_pw_cpu_exception_armv7m) {
188+
if (pw_cpu_exception_ENTRY_BACKEND == dir_pw_cpu_exception_armv7m) {
189189
group_deps += [ "$dir_pw_cpu_exception_armv7m:tests" ]
190190
}
191191

pw_cpu_exception/BUILD

+5-1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,10 @@ licenses(["notice"]) # Apache License 2.0
1919
filegroup(
2020
name = "pw_cpu_exception",
2121
srcs = [
22-
"public/pw_cpu_exception/cpu_exception.h",
22+
"basic_handler.cc",
23+
"public/pw_cpu_exception/entry.h",
24+
"public/pw_cpu_exception/handler.h",
25+
"public/pw_cpu_exception/support.h",
26+
"start_exception_handler.cc",
2327
],
2428
)

pw_cpu_exception/BUILD.gn

+62-4
Original file line numberDiff line numberDiff line change
@@ -22,15 +22,73 @@ config("default_config") {
2222
include_dirs = [ "public" ]
2323
}
2424

25-
pw_facade("pw_cpu_exception") {
26-
backend = pw_cpu_exception_BACKEND
25+
group("pw_cpu_exception") {
26+
public_deps = [
27+
":entry",
28+
":handler",
29+
]
30+
}
31+
32+
# This module has three facades, each of whose backends are set with a
33+
# different GN variable.
34+
#
35+
# - entry: This is the library that handles early exception entry and prepares
36+
# any CPU state that must be available to the exception handler via the
37+
# pw_CpuState object. The backend for this facade will be architecture-
38+
# specific.
39+
# Set this facade's backend via `pw_cpu_exception_ENTRY_BACKEND`
40+
#
41+
# - handler: This facade is backed by an application-specific handler that
42+
# determines what to do when an exception is encountered. This may be
43+
# capturing a crash report before resetting the device, or in some cases
44+
# handling the exception to allow execution to continue.
45+
# Set this facade's backend via `pw_cpu_exception_HANDLER_BACKEND`
46+
#
47+
# - support: This facade provides architecture-independent functions that may be
48+
# helpful for dumping CPU state in various forms. This allows an application
49+
# to create an application-specific handler that is portable across multiple
50+
# architectures.
51+
# Set this facade's backend via `pw_cpu_exception_SUPPORT_BACKEND`
52+
53+
pw_facade("entry") {
54+
backend = pw_cpu_exception_ENTRY_BACKEND
55+
facade_name = "entry_facade"
56+
public_configs = [ ":default_config" ]
57+
public_deps = [ "$dir_pw_preprocessor" ]
58+
deps = [ ":handler_facade" ]
59+
public = [ "public/pw_cpu_exception/entry.h" ]
60+
}
61+
62+
pw_facade("handler") {
63+
backend = pw_cpu_exception_HANDLER_BACKEND
64+
facade_name = "handler_facade"
2765
public_configs = [ ":default_config" ]
2866
public_deps = [
2967
"$dir_pw_preprocessor",
3068
"$dir_pw_span",
31-
"$dir_pw_string",
3269
]
33-
public = [ "public/pw_cpu_exception/cpu_exception.h" ]
70+
sources = [ "start_exception_handler.cc" ]
71+
public = [ "public/pw_cpu_exception/handler.h" ]
72+
}
73+
74+
# This library is technically optional. It is recommended to use `support` when
75+
# doing basic dumps of CPU state. As an alternative, projects may choose to
76+
# directly depend on the entry backend if they require direct access to
77+
# pw_CpuExceptionState members.
78+
pw_facade("support") {
79+
backend = pw_cpu_exception_SUPPORT_BACKEND
80+
facade_name = "support_facade"
81+
public_configs = [ ":default_config" ]
82+
public_deps = [ "$dir_pw_span" ]
83+
public = [ "public/pw_cpu_exception/support.h" ]
84+
}
85+
86+
pw_source_set("basic_handler") {
87+
deps = [
88+
":handler_facade",
89+
dir_pw_log,
90+
]
91+
sources = [ "basic_handler.cc" ]
3492
}
3593

3694
pw_doc_group("docs") {

pw_cpu_exception/backend.gni

+3-1
Original file line numberDiff line numberDiff line change
@@ -14,5 +14,7 @@
1414

1515
declare_args() {
1616
# Backend for the pw_cpu_exception module.
17-
pw_cpu_exception_BACKEND = ""
17+
pw_cpu_exception_ENTRY_BACKEND = ""
18+
pw_cpu_exception_HANDLER_BACKEND = ""
19+
pw_cpu_exception_SUPPORT_BACKEND = ""
1820
}

pw_cpu_exception/basic_handler.cc

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
// Copyright 2020 The Pigweed Authors
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4+
// use this file except in compliance with the License. You may obtain a copy of
5+
// the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11+
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12+
// License for the specific language governing permissions and limitations under
13+
// the License.
14+
#include <cstdlib>
15+
16+
#include "pw_cpu_exception/handler.h"
17+
#include "pw_log/log.h"
18+
19+
namespace pw::cpu_exception {
20+
21+
extern "C" void pw_CpuExceptionDefaultHandler(pw_CpuExceptionState*) {
22+
PW_LOG_CRITICAL("Unhandled CPU exception encountered!");
23+
// TODO(pwbug/95): Replace with pw_abort when that module exists.
24+
std::abort();
25+
}
26+
27+
} // namespace pw::cpu_exception

pw_cpu_exception/docs.rst

+31-20
Original file line numberDiff line numberDiff line change
@@ -22,8 +22,8 @@ called immediately upon a CPU exception. For specifics on how this may be done,
2222
see the backend documentation for your architecture.
2323

2424
Applications must also provide an implementation for
25-
``pw::cpu_exception::HandleCpuException()``. The behavior of this functions
26-
is entirely up to the application/project, but some examples are provided below:
25+
``pw_CpuExceptionDefaultHandler()``. The behavior of this functions is entirely
26+
up to the application/project, but some examples are provided below:
2727

2828
* Enter an infinite loop so the device can be debugged by JTAG.
2929
* Reset the device.
@@ -35,42 +35,53 @@ is entirely up to the application/project, but some examples are provided below:
3535
Module Usage
3636
============
3737
Basic usage of this module entails applications supplying a definition for
38-
``pw::cpu_exception::HandleCpuException()``. ``HandleCpuException()`` should
39-
contain any logic to determine if a exception can be recovered from, as well
40-
as necessary actions to properly recover. If the device cannot recover from the
38+
``pw_CpuExceptionDefaultHandler()``. ``pw_CpuExceptionDefaultHandler()`` should
39+
contain any logic to determine if a exception can be recovered from, as well as
40+
necessary actions to properly recover. If the device cannot recover from the
4141
exception, the function should **not** return.
4242

43+
``pw_CpuExceptionDefaultHandler()`` is called indirectly, and may be overridden
44+
at runtime via ``pw_CpuExceptionSetHandler()``. The handler can also be reset to
45+
point to ``pw_CpuExceptionDefaultHandler()`` by calling
46+
``pw_CpuExceptionRestoreDefaultHandler()``.
47+
4348
When writing an exception handler, prefer to use the functions provided by this
44-
interface rather than relying on the backend implementation of ``CpuState``.
45-
This allows better code portability as it helps prevent an application fault
46-
handler from being tied to a single backend.
49+
interface rather than relying on the backend implementation of
50+
``pw_CpuExceptionState``. This allows better code portability as it helps
51+
prevent an application fault handler from being tied to a single backend.
4752

4853
For example; when logging or dumping CPU state, prefer ``ToString()`` or
49-
``RawFaultingCpuState()`` over directly accessing members of a ``CpuState``
50-
object.
54+
``RawFaultingCpuState()`` over directly accessing members of a
55+
``pw_CpuExceptionState`` object.
5156

5257
Some exception handling behavior may require architecture-specific CPU state to
5358
attempt to correct a fault. In this situation, the application's exception
5459
handler will be tied to the backend implementation of the CPU exception module.
5560

56-
Dependencies
57-
============
58-
* ``pw_span``
59-
* ``pw_preprocessor``
60-
6161
Backend Expectations
6262
====================
6363
CPU exception backends do not provide an exception handler, but instead provide
6464
mechanisms to capture CPU state for use by an application's exception handler,
6565
and allow recovery from CPU exceptions when possible.
6666

67-
* A backend should provide a definition for the ``CpuState`` struct that
68-
provides suitable means to access and modify any captured CPU state.
67+
* A backend should provide a definition for the ``pw_CpuExceptionState``
68+
struct that provides suitable means to access and modify any captured CPU
69+
state.
6970
* If an application's exception handler modifies the captured CPU state, the
7071
state should be treated as though it were the original state of the CPU when
7172
the exception occurred. The backend may need to manually restore some of the
7273
modified state to ensure this on exception handler return.
7374
* A backend should implement the ``pw_CpuExceptionEntry()`` function that will
74-
call ``HandleCpuException()`` after performing any necessary actions prior
75-
to handing control to the application's exception handler (e.g. capturing
76-
necessary CPU state).
75+
call ``pw_HandleCpuException()`` after performing any necessary
76+
actions prior to handing control to the application's exception handler
77+
(e.g. capturing necessary CPU state).
78+
79+
Compatibility
80+
=============
81+
Most of the pw_cpu_exception module is C-compatible. The exception to this is
82+
the "support" facade and library, which requires C++.
83+
84+
Dependencies
85+
============
86+
* ``pw_span``
87+
* ``pw_preprocessor``

pw_cpu_exception/public/pw_cpu_exception/cpu_exception.h

-78
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
// Copyright 2019 The Pigweed Authors
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4+
// use this file except in compliance with the License. You may obtain a copy of
5+
// the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11+
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12+
// License for the specific language governing permissions and limitations under
13+
// the License.
14+
#pragma once
15+
16+
// Platform-independent mechanism to catch hardware CPU faults in user code.
17+
// This module encapsulates low level CPU exception handling assembly for the
18+
// platform. By default, this module invokes the following user-defined function
19+
// after early exception handling completes:
20+
//
21+
// pw_CpuExceptionDefaultHandler(pw_CpuExceptionState* state)
22+
//
23+
// If platform-dependent access to the CPU registers is needed, then
24+
// applications can include the respective backend module directly; for example
25+
// cpu_exception_armv7m.
26+
//
27+
// IMPORTANT: To use this module, you MUST implement
28+
// pw_CpuExceptionDefaultHandler() in some part of your application.
29+
30+
#include "pw_preprocessor/compiler.h"
31+
#include "pw_preprocessor/util.h"
32+
33+
// Low-level raw exception entry handler.
34+
//
35+
// Captures faulting CPU state into a platform-specific pw_CpuExceptionState
36+
// object, then calls the user-provided fault handler.
37+
//
38+
// This function should be called immediately after a fault; typically by being
39+
// in the interrupt vector table entries for the hard fault exceptions.
40+
//
41+
// Note: applications should almost never invoke this directly; if you do, make
42+
// sure you know what you are doing.
43+
PW_EXTERN_C PW_NO_PROLOGUE void pw_CpuExceptionEntry(void);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
// Copyright 2020 The Pigweed Authors
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4+
// use this file except in compliance with the License. You may obtain a copy of
5+
// the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11+
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12+
// License for the specific language governing permissions and limitations under
13+
// the License.
14+
#pragma once
15+
16+
#include "pw_preprocessor/compiler.h"
17+
#include "pw_preprocessor/util.h"
18+
19+
PW_EXTERN_C_START
20+
21+
// Forward declaration of pw_CpuExceptionState. Definition provided by cpu
22+
// exception entry backend.
23+
struct pw_CpuExceptionState;
24+
25+
// By default, the exception entry function will terminate by handing execution
26+
// over to pw_CpuExceptionDefaultHandler(). This can be used to override the
27+
// current handler. This allows runtime insertion of an exception handler which
28+
// may also be helpful for loading a bootloader exception handler by default
29+
// that an application overrides.
30+
void pw_CpuExceptionSetHandler(void (*handler)(pw_CpuExceptionState*));
31+
32+
// Set the exception handler to point to pw_CpuExceptionDefaultHandler().
33+
void pw_CpuExceptionRestoreDefaultHandler(void);
34+
35+
// Application-defined recoverable CPU exception handler.
36+
//
37+
// Applications must define this function; it is not defined by the exception
38+
// entry backend. After CPU state is captured by the cpu exception entry
39+
// backend, this function is called. Applications can then choose to either
40+
// gracefully handle the exception and return, or decide the exception cannot be
41+
// handled and abort normal execution (e.g. reset).
42+
//
43+
// Examples of what applications could do in the handler: gracefully recover
44+
// (e.g. enabling a floating point unit after triggering an exception executing
45+
// a floating point instruction), reset the device, or wait for a debugger to
46+
// attach.
47+
//
48+
// See the cpu_exception module documentation for more details.
49+
PW_USED void pw_CpuExceptionDefaultHandler(pw_CpuExceptionState* state);
50+
51+
// This is the underlying function the CPU exception entry backend should call.
52+
// This calls the currently set handler.
53+
void pw_HandleCpuException(void* cpu_state);
54+
55+
PW_EXTERN_C_END

0 commit comments

Comments
 (0)