/* * Copyright 2014 Google Inc. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "checked_adder.h" #include #include class CheckedAdder : public Adder { private: bool add_overflows(int x, int y) { if (y > x) std::swap(x, y); // Now y <= x. const int half_max = INT_MAX / 2; const int half_min = INT_MIN / 2; if (x > half_max) { // We can't have negative overflow, but might have positive overflow. if (y > half_max) return true; if (y <= 0) return false; // x <= INT_MAX && y <= half_max, // so: x + y <= INT_MAX + half_max // so: x - half_max + y <= INT_MAX // so: (x - half_max + y) doesn't overflow. // (x + y) > INT_MAX iff (x - half_max + y) > (INT_MAX - half_max) return (x - half_max + y) > (INT_MAX - half_max); } // y <= x <= half_max, can't have positive overflow. if (y < half_min) { // We can't have positive overflow, but might have negative overflow. if (x < half_min) return true; if (x >= 0) return false; // y >= INT_MIN && x >= half_min, // so: y + x >= INT_MIN + half_min // so: y - half_min + x >= INT_MAX // so: (y - half_min + x) doesn't overflow. // (y + x) < INT_MIN iff (y - half_min + x) < (INT_MIN - half_min) return (y - half_min + x) < (INT_MIN - half_min); } // Neither negative nor positive overflow. return false; } public: INJECT(CheckedAdder()) = default; virtual int add(int x, int y) override { if (add_overflows(x, y)) { std::cerr << "CheckedAdder: detected overflow during addition of " << x << " and " << y << std::endl; abort(); } return x + y; } }; fruit::Component getCheckedAdderComponent() { return fruit::createComponent().bind(); }