xref: /aosp_15_r20/external/google-fruit/examples/simple_injection/checked_adder.cpp (revision a65addddcf69f38db5b288d787b6b7571a57bb8f)
1*a65addddSAndroid Build Coastguard Worker /*
2*a65addddSAndroid Build Coastguard Worker  * Copyright 2014 Google Inc. All rights reserved.
3*a65addddSAndroid Build Coastguard Worker  *
4*a65addddSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*a65addddSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*a65addddSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*a65addddSAndroid Build Coastguard Worker  *
8*a65addddSAndroid Build Coastguard Worker  *     http://www.apache.org/licenses/LICENSE-2.0
9*a65addddSAndroid Build Coastguard Worker  *
10*a65addddSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*a65addddSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*a65addddSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*a65addddSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*a65addddSAndroid Build Coastguard Worker  * limitations under the License.
15*a65addddSAndroid Build Coastguard Worker  */
16*a65addddSAndroid Build Coastguard Worker 
17*a65addddSAndroid Build Coastguard Worker #include "checked_adder.h"
18*a65addddSAndroid Build Coastguard Worker 
19*a65addddSAndroid Build Coastguard Worker #include <climits>
20*a65addddSAndroid Build Coastguard Worker #include <iostream>
21*a65addddSAndroid Build Coastguard Worker 
22*a65addddSAndroid Build Coastguard Worker class CheckedAdder : public Adder {
23*a65addddSAndroid Build Coastguard Worker private:
add_overflows(int x,int y)24*a65addddSAndroid Build Coastguard Worker   bool add_overflows(int x, int y) {
25*a65addddSAndroid Build Coastguard Worker     if (y > x)
26*a65addddSAndroid Build Coastguard Worker       std::swap(x, y);
27*a65addddSAndroid Build Coastguard Worker     // Now y <= x.
28*a65addddSAndroid Build Coastguard Worker     const int half_max = INT_MAX / 2;
29*a65addddSAndroid Build Coastguard Worker     const int half_min = INT_MIN / 2;
30*a65addddSAndroid Build Coastguard Worker     if (x > half_max) {
31*a65addddSAndroid Build Coastguard Worker       // We can't have negative overflow, but might have positive overflow.
32*a65addddSAndroid Build Coastguard Worker       if (y > half_max)
33*a65addddSAndroid Build Coastguard Worker         return true;
34*a65addddSAndroid Build Coastguard Worker       if (y <= 0)
35*a65addddSAndroid Build Coastguard Worker         return false;
36*a65addddSAndroid Build Coastguard Worker       // x <= INT_MAX && y <= half_max,
37*a65addddSAndroid Build Coastguard Worker       // so: x + y <= INT_MAX + half_max
38*a65addddSAndroid Build Coastguard Worker       // so: x - half_max + y <= INT_MAX
39*a65addddSAndroid Build Coastguard Worker       // so: (x - half_max + y) doesn't overflow.
40*a65addddSAndroid Build Coastguard Worker       // (x + y) > INT_MAX iff (x - half_max + y) > (INT_MAX - half_max)
41*a65addddSAndroid Build Coastguard Worker       return (x - half_max + y) > (INT_MAX - half_max);
42*a65addddSAndroid Build Coastguard Worker     }
43*a65addddSAndroid Build Coastguard Worker     // y <= x <= half_max, can't have positive overflow.
44*a65addddSAndroid Build Coastguard Worker     if (y < half_min) {
45*a65addddSAndroid Build Coastguard Worker       // We can't have positive overflow, but might have negative overflow.
46*a65addddSAndroid Build Coastguard Worker       if (x < half_min)
47*a65addddSAndroid Build Coastguard Worker         return true;
48*a65addddSAndroid Build Coastguard Worker       if (x >= 0)
49*a65addddSAndroid Build Coastguard Worker         return false;
50*a65addddSAndroid Build Coastguard Worker       // y >= INT_MIN && x >= half_min,
51*a65addddSAndroid Build Coastguard Worker       // so: y + x >= INT_MIN + half_min
52*a65addddSAndroid Build Coastguard Worker       // so: y - half_min + x >= INT_MAX
53*a65addddSAndroid Build Coastguard Worker       // so: (y - half_min + x) doesn't overflow.
54*a65addddSAndroid Build Coastguard Worker       // (y + x) < INT_MIN iff (y - half_min + x) < (INT_MIN - half_min)
55*a65addddSAndroid Build Coastguard Worker       return (y - half_min + x) < (INT_MIN - half_min);
56*a65addddSAndroid Build Coastguard Worker     }
57*a65addddSAndroid Build Coastguard Worker     // Neither negative nor positive overflow.
58*a65addddSAndroid Build Coastguard Worker     return false;
59*a65addddSAndroid Build Coastguard Worker   }
60*a65addddSAndroid Build Coastguard Worker 
61*a65addddSAndroid Build Coastguard Worker public:
62*a65addddSAndroid Build Coastguard Worker   INJECT(CheckedAdder()) = default;
63*a65addddSAndroid Build Coastguard Worker 
add(int x,int y)64*a65addddSAndroid Build Coastguard Worker   virtual int add(int x, int y) override {
65*a65addddSAndroid Build Coastguard Worker     if (add_overflows(x, y)) {
66*a65addddSAndroid Build Coastguard Worker       std::cerr << "CheckedAdder: detected overflow during addition of " << x << " and " << y << std::endl;
67*a65addddSAndroid Build Coastguard Worker       abort();
68*a65addddSAndroid Build Coastguard Worker     }
69*a65addddSAndroid Build Coastguard Worker     return x + y;
70*a65addddSAndroid Build Coastguard Worker   }
71*a65addddSAndroid Build Coastguard Worker };
72*a65addddSAndroid Build Coastguard Worker 
getCheckedAdderComponent()73*a65addddSAndroid Build Coastguard Worker fruit::Component<Adder> getCheckedAdderComponent() {
74*a65addddSAndroid Build Coastguard Worker   return fruit::createComponent().bind<Adder, CheckedAdder>();
75*a65addddSAndroid Build Coastguard Worker }
76