blob: b6f10c87596edfa3ce448e563cc2ea3526367bc0 [file] [log] [blame]
Steven Morelanddea3cf92019-07-16 18:06:55 -07001/*
2 * Copyright (C) 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Steven Morelanddea3cf92019-07-16 18:06:55 -070016#include <binder/Stability.h>
17
18namespace android {
19namespace internal {
20
21void Stability::markCompilationUnit(IBinder* binder) {
Steven Moreland2a9f32f2019-07-31 17:51:25 -070022 status_t result = set(binder, kLocalStability, true /*log*/);
Steven Morelanddea3cf92019-07-16 18:06:55 -070023 LOG_ALWAYS_FATAL_IF(result != OK, "Should only mark known object.");
24}
25
26void Stability::markVintf(IBinder* binder) {
Steven Moreland2a9f32f2019-07-31 17:51:25 -070027 status_t result = set(binder, Level::VINTF, true /*log*/);
Steven Morelanddea3cf92019-07-16 18:06:55 -070028 LOG_ALWAYS_FATAL_IF(result != OK, "Should only mark known object.");
29}
30
Steven Moreland64ae9172019-08-02 20:45:15 -070031void Stability::debugLogStability(const std::string& tag, const sp<IBinder>& binder) {
32 ALOGE("%s: stability is %s", tag.c_str(), stabilityString(get(binder.get())).c_str());
33}
34
Steven Morelandc709dd82019-08-05 20:30:14 -070035void Stability::markVndk(IBinder* binder) {
36 status_t result = set(binder, Level::VENDOR, true /*log*/);
37 LOG_ALWAYS_FATAL_IF(result != OK, "Should only mark known object.");
38}
39
Steven Moreland2a9f32f2019-07-31 17:51:25 -070040void Stability::tryMarkCompilationUnit(IBinder* binder) {
41 (void) set(binder, kLocalStability, false /*log*/);
42}
43
44status_t Stability::set(IBinder* binder, int32_t stability, bool log) {
Steven Morelanddea3cf92019-07-16 18:06:55 -070045 Level currentStability = get(binder);
46
47 // null binder is always written w/ 'UNDECLARED' stability
48 if (binder == nullptr) {
49 if (stability == UNDECLARED) {
50 return OK;
51 } else {
Steven Moreland2a9f32f2019-07-31 17:51:25 -070052 if (log) {
53 ALOGE("Null binder written with stability %s.",
54 stabilityString(stability).c_str());
55 }
Steven Morelanddea3cf92019-07-16 18:06:55 -070056 return BAD_TYPE;
57 }
58 }
59
60 if (!isDeclaredStability(stability)) {
Steven Moreland2a9f32f2019-07-31 17:51:25 -070061 if (log) {
Steven Morelanddea3cf92019-07-16 18:06:55 -070062 ALOGE("Can only set known stability, not %d.", stability);
Steven Morelanddea3cf92019-07-16 18:06:55 -070063 }
Steven Moreland2a9f32f2019-07-31 17:51:25 -070064 return BAD_TYPE;
Steven Morelanddea3cf92019-07-16 18:06:55 -070065 }
66
67 if (currentStability != Level::UNDECLARED && currentStability != stability) {
Steven Moreland2a9f32f2019-07-31 17:51:25 -070068 if (log) {
69 ALOGE("Interface being set with %s but it is already marked as %s.",
Steven Moreland732de212019-08-02 20:41:10 -070070 stabilityString(stability).c_str(), stabilityString(currentStability).c_str());
Steven Moreland2a9f32f2019-07-31 17:51:25 -070071 }
Steven Morelanddea3cf92019-07-16 18:06:55 -070072 return BAD_TYPE;
73 }
74
75 if (currentStability == stability) return OK;
76
77 binder->attachObject(
78 reinterpret_cast<void*>(&Stability::get),
79 reinterpret_cast<void*>(stability),
80 nullptr /*cleanupCookie*/,
81 nullptr /*cleanup function*/);
82
83 return OK;
84}
85
86Stability::Level Stability::get(IBinder* binder) {
87 if (binder == nullptr) return UNDECLARED;
88
89 return static_cast<Level>(reinterpret_cast<intptr_t>(
90 binder->findObject(reinterpret_cast<void*>(&Stability::get))));
91}
92
93bool Stability::check(int32_t provided, Level required) {
94 bool stable = (provided & required) == required;
95
96 if (!isDeclaredStability(provided) && provided != UNDECLARED) {
97 ALOGE("Unknown stability when checking interface stability %d.", provided);
98
99 stable = false;
100 }
101
102 if (!stable) {
Steven Morelandc709dd82019-08-05 20:30:14 -0700103 ALOGE("Cannot do a user transaction on a %s binder in a %s context.",
104 stabilityString(provided).c_str(),
105 stabilityString(required).c_str());
Steven Morelanddea3cf92019-07-16 18:06:55 -0700106 }
107
108 return stable;
109}
110
111bool Stability::isDeclaredStability(int32_t stability) {
112 return stability == VENDOR || stability == SYSTEM || stability == VINTF;
113}
114
115std::string Stability::stabilityString(int32_t stability) {
116 switch (stability) {
117 case Level::UNDECLARED: return "undeclared stability";
118 case Level::VENDOR: return "vendor stability";
119 case Level::SYSTEM: return "system stability";
120 case Level::VINTF: return "vintf stability";
121 }
122 return "unknown stability " + std::to_string(stability);
123}
124
125} // namespace internal
126} // namespace stability