File: ErrorObjectCommon.cpp

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (79 lines) | stat: -rw-r--r-- 3,162 bytes parent folder | download
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
//===--- ErrorObjectCommon.cpp - Recoverable error object -----------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2019 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This implements the parts of the standard Error protocol type which are
// shared between the ObjC-interoperable implementation and the native
// implementation. The parts specific to each implementation can be found in
// ErrorObject.mm (for the ObjC-interoperable parts) and ErrorObjectNative.cpp.
//
//===----------------------------------------------------------------------===//

#include "swift/Runtime/Concurrent.h"
#include "swift/Runtime/Config.h"
#include "ErrorObject.h"
#include "ErrorObjectTestSupport.h"

using namespace swift;

std::atomic<void (*)(SwiftError *error)> swift::_swift_willThrow;

void swift::_swift_setWillThrowHandler(void (* handler)(SwiftError *error)) {
  _swift_willThrow.store(handler, std::memory_order_release);
}

/// Breakpoint hook for debuggers that is called for untyped throws, and
/// calls _swift_willThrow if set.
SWIFT_CC(swift) void
swift::swift_willThrow(SWIFT_CONTEXT void *unused,
                       SWIFT_ERROR_RESULT SwiftError **error) {
  // Cheap check to bail out early, since we expect there to be no callbacks
  // the vast majority of the time.
  auto handler = _swift_willThrow.load(std::memory_order_acquire);
  if (SWIFT_UNLIKELY(handler)) {
    (* handler)(*error);
  }
}

std::atomic<void (*)(
  OpaqueValue *value,
  const Metadata *type,
  const WitnessTable *errorConformance
)> swift::_swift_willThrowTypedImpl;

/// Breakpoint hook for debuggers that is called for typed throws, and calls
/// _swift_willThrowTypedImpl if set. If not set and _swift_willThrow is set, this calls
/// that hook instead and implicitly boxes the typed error in an any Error for that call.
SWIFT_CC(swift) void
swift::swift_willThrowTypedImpl(OpaqueValue *value,
                                const Metadata *type,
                                const WitnessTable *errorConformance) {
  // Cheap check to bail out early, since we expect there to be no callbacks
  // the vast majority of the time.
  auto handler = _swift_willThrowTypedImpl.load(std::memory_order_acquire);
  if (SWIFT_UNLIKELY(handler)) {
    (* handler)(value, type, errorConformance);
  } else {
    auto fallbackHandler = _swift_willThrow.load(std::memory_order_acquire);
    if (SWIFT_UNLIKELY(fallbackHandler)) {
      // Form an error box containing the error.
      BoxPair boxedError = swift_allocError(
        type, errorConformance, value, /*isTake=*/false);

      // Hand the boxed error off to the handler.
      auto errorBox = reinterpret_cast<SwiftError *>(boxedError.object);
      (* fallbackHandler)(errorBox);

      // Release the error box.
      swift_errorRelease(errorBox);
    }
  }
}