23#include "absl/algorithm/container.h"
24#include "absl/container/btree_set.h"
25#include "absl/container/inlined_vector.h"
26#include "absl/log/check.h"
27#include "absl/strings/str_format.h"
28#include "absl/strings/string_view.h"
43int FindSegmentIndex(
const std::vector<PiecewiseSegment>& segments, int64_t x) {
44 if (segments.empty() || segments.front().start_x() > x) {
50 std::vector<PiecewiseSegment>::const_iterator position = std::upper_bound(
52 if (position == segments.end()) {
53 return segments.size() - 1;
55 position -= position->start_x() >
x ? 1 : 0;
57 return position - segments.begin();
60inline bool IsAtBounds(int64_t value) {
64inline bool PointInsideRange(int64_t point, int64_t range_start,
66 return range_start <= point && range_end >= point;
73 return right.slope() >= left.slope() && right.start_x() == left.end_x() &&
74 right.start_y() == left.end_y();
77uint64_t UnsignedCapAdd(uint64_t left, uint64_t right) {
81uint64_t UnsignedCapProd(uint64_t left, uint64_t right) {
82 if (right == 0)
return 0;
90 int64_t
slope, int64_t other_point_x)
91 : slope_(
slope), reference_x_(point_x), reference_y_(point_y) {
92 start_x_ = std::min(point_x, other_point_x);
93 end_x_ = std::max(point_x, other_point_x);
95 reference_x_ < 0 ? SafeValuePostReference(0) : SafeValuePreReference(0);
99 CHECK_GE(x, start_x_);
102 const int64_t span_x =
CapSub(x, reference_x_);
105 return SafeValuePostReference(x);
108 return SafeValuePreReference(x);
111 const int64_t span_y =
CapProd(slope_, span_x);
112 if (IsAtBounds(span_y)) {
114 return SafeValuePostReference(x);
116 return SafeValuePreReference(x);
120 const int64_t value =
CapAdd(reference_y_, span_y);
121 if (IsAtBounds(value)) {
123 return SafeValuePostReference(x);
125 return SafeValuePreReference(x);
132int64_t PiecewiseSegment::SafeValuePostReference(int64_t x)
const {
133 DCHECK_GE(x, reference_x_);
134 const uint64_t span_x =
static_cast<uint64_t
>(x) - reference_x_;
141 }
else if (slope_ > 0) {
143 const uint64_t span_y = UnsignedCapProd(span_x, slope_);
144 if (reference_y_ == 0) {
146 }
else if (reference_y_ > 0) {
147 const uint64_t unsigned_sum = UnsignedCapAdd(reference_y_, span_y);
149 :
static_cast<int64_t
>(unsigned_sum);
151 const uint64_t opp_reference_y = -
static_cast<uint64_t
>(reference_y_);
152 if (span_y >= opp_reference_y) {
153 return span_y - opp_reference_y >
kint64max
155 :
static_cast<int64_t
>(span_y - opp_reference_y);
157 return opp_reference_y - span_y >
static_cast<uint64_t
>(
kint64max) + 1
159 : -static_cast<int64_t>(opp_reference_y - span_y);
164 const uint64_t span_y = UnsignedCapProd(span_x, -slope_);
165 if (reference_y_ == 0) {
167 }
else if (reference_y_ < 0) {
168 const uint64_t opp_reference_y = -
static_cast<uint64_t
>(reference_y_);
169 const uint64_t opp_unsigned_sum = UnsignedCapAdd(opp_reference_y, span_y);
172 : -
static_cast<int64_t
>(opp_unsigned_sum);
174 if (reference_y_ >= span_y) {
177 :
static_cast<int64_t
>(reference_y_ - span_y);
179 return span_y - reference_y_ >
static_cast<uint64_t
>(
kint64max) + 1
181 : -static_cast<int64_t>(span_y - reference_y_);
187int64_t PiecewiseSegment::SafeValuePreReference(int64_t x)
const {
188 DCHECK_LE(x, reference_x_);
189 const uint64_t span_x =
static_cast<uint64_t
>(reference_x_) - x;
193 }
else if (slope_ > 0) {
195 const uint64_t span_y = UnsignedCapProd(span_x, slope_);
196 if (reference_y_ == 0) {
198 }
else if (reference_y_ > 0) {
199 if (reference_y_ >= span_y) {
202 :
static_cast<int64_t
>(reference_y_ - span_y);
204 return span_y - reference_y_ >
static_cast<uint64_t
>(
kint64max) + 1
206 : -static_cast<uint64_t>(span_y - reference_y_);
209 const uint64_t opp_reference_y = -
static_cast<uint64_t
>(reference_y_);
210 const uint64_t opp_unsigned_sum = UnsignedCapAdd(opp_reference_y, span_y);
213 : -
static_cast<uint64_t
>(opp_unsigned_sum);
217 const uint64_t span_y = UnsignedCapProd(span_x, -slope_);
218 if (reference_y_ == 0) {
220 }
else if (reference_y_ < 0) {
221 const uint64_t opp_reference_y = -
static_cast<uint64_t
>(reference_y_);
222 if (span_y >= opp_reference_y) {
223 return span_y - opp_reference_y >
kint64max
225 :
static_cast<int64_t
>(span_y - opp_reference_y);
227 return opp_reference_y - span_y >
static_cast<uint64_t
>(
kint64max) + 1
229 : -static_cast<uint64_t>(opp_reference_y - span_y);
232 const uint64_t unsigned_sum = UnsignedCapAdd(reference_y_, span_y);
234 :
static_cast<int64_t
>(unsigned_sum);
241 return segment1.start_x_ < segment2.start_x_;
250 end_x_ = std::max(end_x_,
end_x);
254 if (IsAtBounds(
CapAdd(reference_x_, constant))) {
255 LOG(ERROR) <<
"Segment Overflow: " <<
DebugString();
258 start_x_ =
CapAdd(start_x_, constant);
259 end_x_ =
CapAdd(end_x_, constant);
260 reference_x_ =
CapAdd(reference_x_, constant);
264 if (IsAtBounds(
CapAdd(reference_y_, constant))) {
265 LOG(ERROR) <<
"Segment Overflow: " <<
DebugString();
268 reference_y_ =
CapAdd(reference_y_, constant);
272 std::string result = absl::StrFormat(
273 "PiecewiseSegment(<start: (%d, %d), end: (%d, %d), "
274 "reference: (%d, %d), slope = %d>)",
275 start_x_,
Value(start_x_), end_x_,
Value(end_x_), reference_x_,
276 reference_y_, slope_);
283PiecewiseLinearFunction::PiecewiseLinearFunction(
284 std::vector<PiecewiseSegment> segments)
285 : is_modified_(true),
287 is_non_decreasing_(false),
288 is_non_increasing_(false) {
292 for (
int i = 0; i < segments.size() - 1; ++i) {
294 LOG(FATAL) <<
"Overlapping segments: " << segments[i].DebugString()
295 <<
" & " << segments[i + 1].DebugString();
299 for (
const auto& segment : segments) {
300 InsertSegment(segment);
305 std::vector<int64_t> points_x, std::vector<int64_t> points_y,
306 std::vector<int64_t> slopes, std::vector<int64_t> other_points_x) {
307 CHECK_EQ(points_x.size(), points_y.size());
308 CHECK_EQ(points_x.size(), other_points_x.size());
309 CHECK_EQ(points_x.size(), slopes.size());
310 CHECK_GT(points_x.size(), 0);
312 std::vector<PiecewiseSegment>
segments;
313 for (
int i = 0; i < points_x.size(); ++i) {
318 return new PiecewiseLinearFunction(std::move(
segments));
322 std::vector<int64_t> points_x, std::vector<int64_t> points_y,
323 std::vector<int64_t> other_points_x) {
324 CHECK_EQ(points_x.size(), points_y.size());
325 CHECK_EQ(points_x.size(), other_points_x.size());
326 CHECK_GT(points_x.size(), 0);
328 std::vector<PiecewiseSegment>
segments;
329 for (
int i = 0; i < points_x.size(); ++i) {
334 return new PiecewiseLinearFunction(std::move(
segments));
338 int64_t initial_level, std::vector<int64_t> points_x,
339 std::vector<int64_t> slopes) {
340 CHECK_EQ(points_x.size(), slopes.size() - 1);
341 CHECK_GT(points_x.size(), 0);
343 int64_t level = initial_level;
344 std::vector<PiecewiseSegment>
segments;
348 level = segment.
Value(points_x[0]);
349 for (
int i = 1; i < points_x.size(); ++i) {
353 level = segment.
Value(points_x[i]);
358 return new PiecewiseLinearFunction(std::move(
segments));
362 int64_t point_x, int64_t point_y, int64_t slope, int64_t other_point_x) {
365 std::vector<PiecewiseSegment>
segments = {
367 return new PiecewiseLinearFunction(std::move(
segments));
371 int64_t point_x, int64_t point_y, int64_t slope) {
372 std::vector<PiecewiseSegment>
segments = {
374 return new PiecewiseLinearFunction(std::move(
segments));
378 int64_t point_x, int64_t point_y, int64_t slope) {
379 std::vector<PiecewiseSegment>
segments = {
381 return new PiecewiseLinearFunction(std::move(
segments));
385 int64_t slope, int64_t value) {
386 std::vector<PiecewiseSegment>
segments = {
391 return new PiecewiseLinearFunction(std::move(
segments));
395 int64_t reference, int64_t earliness_slope, int64_t tardiness_slope) {
396 std::vector<PiecewiseSegment>
segments = {
399 CHECK_GE(earliness_slope, 0);
400 CHECK_GE(tardiness_slope, 0);
401 return new PiecewiseLinearFunction(std::move(
segments));
406 int64_t early_slack, int64_t late_slack, int64_t earliness_slope,
407 int64_t tardiness_slope) {
408 std::vector<PiecewiseSegment>
segments = {
413 CHECK_GE(earliness_slope, 0);
414 CHECK_GE(tardiness_slope, 0);
415 return new PiecewiseLinearFunction(std::move(
segments));
419 int index = FindSegmentIndex(segments_, x);
423 if (segments_[index].end_x() < x) {
430 const_cast<PiecewiseLinearFunction*
>(
this)->UpdateStatus();
435 const_cast<PiecewiseLinearFunction*
>(
this)->UpdateStatus();
436 return is_non_decreasing_;
440 const_cast<PiecewiseLinearFunction*
>(
this)->UpdateStatus();
441 return is_non_increasing_;
450 const int index = FindSegmentIndex(segments_, x);
451 return segments_[index].Value(x);
455 int64_t range_end)
const {
457 return Value(range_end);
459 return Value(range_start);
461 int start_segment = -1;
462 int end_segment = -1;
463 if (!FindSegmentIndicesFromRange(range_start, range_end, &start_segment,
467 CHECK_GE(end_segment, start_segment);
471 range_maximum = std::max(
Value(range_start), range_maximum);
474 range_maximum = std::max(
Value(range_end), range_maximum);
477 for (
int i = std::max(0, start_segment); i <= end_segment; ++i) {
478 if (PointInsideRange(segments_[i].start_x(), range_start, range_end)) {
479 range_maximum = std::max(range_maximum, segments_[i].start_y());
481 if (PointInsideRange(segments_[i].end_x(), range_start, range_end)) {
482 range_maximum = std::max(range_maximum, segments_[i].end_y());
485 return range_maximum;
489 int64_t range_end)
const {
491 return Value(range_start);
493 return Value(range_end);
495 int start_segment = -1;
496 int end_segment = -1;
497 if (!FindSegmentIndicesFromRange(range_start, range_end, &start_segment,
501 CHECK_GE(end_segment, start_segment);
505 range_minimum = std::min(
Value(range_start), range_minimum);
508 range_minimum = std::min(
Value(range_end), range_minimum);
511 for (
int i = std::max(0, start_segment); i <= end_segment; ++i) {
512 if (PointInsideRange(segments_[i].start_x(), range_start, range_end)) {
513 range_minimum = std::min(range_minimum, segments_[i].start_y());
515 if (PointInsideRange(segments_[i].end_x(), range_start, range_end)) {
516 range_minimum = std::min(range_minimum, segments_[i].end_y());
519 return range_minimum;
523 return GetMaximum(segments_.front().start_x(), segments_.back().end_x());
527 return GetMinimum(segments_.front().start_x(), segments_.back().end_x());
530std::pair<int64_t, int64_t>
533 int64_t value)
const {
537std::pair<int64_t, int64_t>
540 int64_t value)
const {
545std::pair<int64_t, int64_t> ComputeXFromY(int64_t start_x, int64_t start_y,
546 int64_t slope, int64_t y) {
548 const int64_t delta_y =
CapSub(y, start_y);
549 const int64_t delta_x = delta_y / slope;
550 if ((delta_y >= 0 && slope >= 0) || (delta_y <= 0 && slope <= 0)) {
551 const int64_t delta_x_down = delta_x;
552 const int64_t delta_x_up = delta_y % slope == 0 ? delta_x : delta_x + 1;
553 return {delta_x_down + start_x, delta_x_up + start_x};
555 const int64_t delta_x_down = delta_y % slope == 0 ? delta_x : delta_x - 1;
556 const int64_t delta_x_up = -(-delta_y / slope);
557 return {delta_x_down + start_x, delta_x_up + start_x};
561std::pair<int64_t, int64_t> GetRangeInValueRange(int64_t start_x, int64_t end_x,
562 int64_t start_y, int64_t end_y,
566 if ((start_y > value_max && end_y > value_max) ||
567 (start_y < value_min && end_y < value_min)) {
571 if (start_y <= value_max && end_y <= value_max) {
572 x_range_max = {start_x, end_x};
573 }
else if (start_y <= value_max || end_y <= value_max) {
575 ? ComputeXFromY(end_x, end_y, slope, value_max)
576 : ComputeXFromY(start_x, start_y, slope, value_max);
577 if (end_y <= value_max) {
578 x_range_max = {
x.second, end_x};
580 x_range_max = {start_x,
x.first};
584 if (start_y >= value_min && end_y >= value_min) {
585 x_range_min = {start_x, end_x};
586 }
else if (start_y >= value_min || end_y >= value_min) {
588 ? ComputeXFromY(end_x, end_y, slope, value_min)
589 : ComputeXFromY(start_x, start_y, slope, value_min);
590 if (end_y >= value_min) {
591 x_range_min = {
x.second, end_x};
593 x_range_min = {start_x,
x.first};
596 if (x_range_min.first > x_range_max.second ||
597 x_range_max.first > x_range_min.second) {
600 return {std::max(x_range_min.first, x_range_max.first),
601 std::min(x_range_min.second, x_range_max.second)};
605std::pair<int64_t, int64_t>
609 int64_t value_max)
const {
612 int start_segment = -1;
613 int end_segment = -1;
614 if (!FindSegmentIndicesFromRange(range_start, range_end, &start_segment,
616 return {reduced_range_start, reduced_range_end};
618 for (
int i = std::max(0, start_segment); i <= end_segment; ++i) {
619 const auto& segment = segments_[i];
620 const int64_t start_x = std::max(range_start, segment.start_x());
621 const int64_t end_x = std::min(range_end, segment.end_x());
622 const int64_t start_y = segment.Value(start_x);
623 const int64_t end_y = segment.Value(end_x);
624 const std::pair<int64_t, int64_t> range = GetRangeInValueRange(
625 start_x, end_x, start_y, end_y, segment.slope(), value_min, value_max);
626 reduced_range_start = std::min(reduced_range_start, range.first);
627 reduced_range_end = std::max(reduced_range_end, range.second);
629 return {reduced_range_start, reduced_range_end};
634 for (
int i = 0; i < segments_.size(); ++i) {
635 segments_[i].AddConstantToX(constant);
641 for (
int i = 0; i < segments_.size(); ++i) {
642 segments_[i].AddConstantToY(constant);
647 Operation(other, [](int64_t a, int64_t b) {
return CapAdd(a, b); });
651 Operation(other, [](int64_t a, int64_t b) {
return CapSub(a, b); });
654std::vector<PiecewiseLinearFunction*>
656 CHECK_GE(segments_.size(), 1);
658 return {
new PiecewiseLinearFunction(segments_)};
661 std::vector<PiecewiseLinearFunction*> convex_functions;
662 std::vector<PiecewiseSegment> convex_segments;
665 if (convex_segments.empty()) {
666 convex_segments.push_back(segment);
671 if (FormConvexPair(last, segment)) {
673 convex_segments.push_back(segment);
675 convex_functions.push_back(
new PiecewiseLinearFunction(convex_segments));
676 convex_segments.clear();
677 convex_segments.push_back(segment);
681 if (!convex_segments.empty()) {
682 convex_functions.push_back(
683 new PiecewiseLinearFunction(std::move(convex_segments)));
685 return convex_functions;
689 std::string result =
"PiecewiseLinearFunction(";
690 for (
int i = 0; i < segments_.size(); ++i) {
697void PiecewiseLinearFunction::InsertSegment(
const PiecewiseSegment& segment) {
700 if (segments_.empty() || segments_.back().end_x() < segment.
start_x()) {
701 segments_.push_back(segment);
706 if (segments_.back().end_x() == segment.
start_x()) {
707 if (segments_.back().end_y() == segment.
start_y() &&
708 segments_.back().slope() == segment.
slope()) {
709 segments_.back().ExpandEnd(segment.
end_x());
712 segments_.push_back(segment);
716void PiecewiseLinearFunction::Operation(
718 const std::function<int64_t(int64_t, int64_t)>& operation) {
720 std::vector<PiecewiseSegment> own_segments;
721 const std::vector<PiecewiseSegment>& other_segments = other.segments();
722 own_segments.swap(segments_);
724 absl::btree_set<int64_t> start_x_points;
725 for (
int i = 0;
i < own_segments.size(); ++
i) {
726 start_x_points.insert(own_segments[i].start_x());
728 for (
int i = 0;
i < other_segments.size(); ++
i) {
729 start_x_points.insert(other_segments[i].start_x());
732 for (int64_t start_x : start_x_points) {
733 const int own_index = FindSegmentIndex(own_segments, start_x);
734 const int other_index = FindSegmentIndex(other_segments, start_x);
735 if (own_index >= 0 && other_index >= 0) {
736 const PiecewiseSegment& own_segment = own_segments[own_index];
737 const PiecewiseSegment& other_segment = other_segments[other_index];
739 const int64_t end_x =
740 std::min(own_segment.end_x(), other_segment.end_x());
741 const int64_t start_y =
742 operation(own_segment.Value(start_x), other_segment.Value(start_x));
743 const int64_t end_y =
744 operation(own_segment.Value(end_x), other_segment.Value(end_x));
745 const int64_t slope =
746 operation(own_segment.slope(), other_segment.slope());
748 int64_t point_x, point_y, other_point_x;
749 if (IsAtBounds(start_y)) {
752 other_point_x = start_x;
756 other_point_x = end_x;
758 InsertSegment(PiecewiseSegment(point_x, point_y, slope, other_point_x));
763bool PiecewiseLinearFunction::FindSegmentIndicesFromRange(
764 int64_t range_start, int64_t range_end,
int* start_segment,
765 int* end_segment)
const {
766 *start_segment = FindSegmentIndex(segments_, range_start);
767 *end_segment = FindSegmentIndex(segments_, range_end);
768 if (*start_segment == *end_segment) {
769 if (*start_segment < 0) {
773 if (segments_[*start_segment].end_x() < range_start) {
781bool PiecewiseLinearFunction::IsConvexInternal()
const {
782 for (
int i = 1;
i < segments_.size(); ++
i) {
783 if (!FormConvexPair(segments_[i - 1], segments_[i])) {
790bool PiecewiseLinearFunction::IsNonDecreasingInternal()
const {
792 for (
const auto& segment : segments_) {
793 const int64_t start_y = segment.
start_y();
794 const int64_t end_y = segment.
end_y();
795 if (end_y < start_y || start_y < value)
return false;
801bool PiecewiseLinearFunction::IsNonIncreasingInternal()
const {
803 for (
const auto& segment : segments_) {
804 const int64_t start_y = segment.
start_y();
805 const int64_t end_y = segment.
end_y();
806 if (end_y > start_y || start_y > value)
return false;
816 absl::InlinedVector<int64_t, 8>
x_anchors,
817 absl::InlinedVector<int64_t, 8>
y_anchors)
819 DCHECK(absl::c_is_sorted(x_anchors_));
820 DCHECK_EQ(x_anchors_.size(), y_anchors_.size());
821 DCHECK_NE(x_anchors_.size(), 1);
825 absl::string_view line_prefix)
const {
826 if (x_anchors_.size() <= 10) {
827 return "{ " +
DUMP_VARS(x_anchors_, y_anchors_).str() +
"}";
829 return absl::StrFormat(
"{\n%s%s\n%s%s\n}", line_prefix,
830 DUMP_VARS(x_anchors_).str(), line_prefix,
836 const int segment_index = GetSegmentIndex(x);
838 return GetValueOnSegment(x, segment_index);
842 if (x_anchors_.empty())
return kNoValue;
845 if (x <= x_anchors_[0]) {
847 }
else if (x >= x_anchors_.back()) {
848 segment_index = x_anchors_.size() - 2;
850 segment_index = GetSegmentIndex(x);
853 return GetValueOnSegment(x, segment_index);
856int64_t FloatSlopePiecewiseLinearFunction::GetValueOnSegment(
857 int64_t x,
int segment_index)
const {
858 DCHECK_GE(segment_index, 0);
859 DCHECK_LE(segment_index, x_anchors_.size() - 2);
861 static_cast<double>(y_anchors_[segment_index + 1] -
862 y_anchors_[segment_index]) /
863 (x_anchors_[segment_index + 1] - x_anchors_[segment_index]);
865 y_anchors_[segment_index]);
int64_t ComputeConvexValue(int64_t x) const
FloatSlopePiecewiseLinearFunction()=default
int64_t ComputeInBoundsValue(int64_t x) const
std::string DebugString(absl::string_view line_prefix={}) const
const absl::InlinedVector< int64_t, 8 > & x_anchors() const
const absl::InlinedVector< int64_t, 8 > & y_anchors() const
static const int kNoValue
FloatSlopePiecewiseLinearFunction.
static IntOut Round(FloatIn x)
int64_t GetMaximum() const
Returns the maximum value of all the segments in the function.
std::vector< PiecewiseLinearFunction * > DecomposeToConvexFunctions() const
static PiecewiseLinearFunction * CreateFullDomainFunction(int64_t initial_level, std::vector< int64_t > points_x, std::vector< int64_t > slopes)
static PiecewiseLinearFunction * CreateRightRayFunction(int64_t point_x, int64_t point_y, int64_t slope)
bool IsNonIncreasing() const
Returns true if the piecewise linear function is non-increasing.
std::pair< int64_t, int64_t > GetSmallestRangeLessThanValue(int64_t range_start, int64_t range_end, int64_t value) const
static const int kNotFound
PiecewiseLinearFunction.
static PiecewiseLinearFunction * CreateFixedChargeFunction(int64_t slope, int64_t value)
std::string DebugString() const
void Add(const PiecewiseLinearFunction &other)
static PiecewiseLinearFunction * CreateStepFunction(std::vector< int64_t > points_x, std::vector< int64_t > points_y, std::vector< int64_t > other_points_x)
int64_t GetMinimum() const
Returns the minimum value of all the segments in the function.
void AddConstantToY(int64_t constant)
const std::vector< PiecewiseSegment > & segments() const
static PiecewiseLinearFunction * CreateOneSegmentFunction(int64_t point_x, int64_t point_y, int64_t slope, int64_t other_point_x)
Builds a function consisting of one segment.
std::pair< int64_t, int64_t > GetSmallestRangeInValueRange(int64_t range_start, int64_t range_end, int64_t value_min, int64_t value_max) const
static PiecewiseLinearFunction * CreateLeftRayFunction(int64_t point_x, int64_t point_y, int64_t slope)
bool IsNonDecreasing() const
Returns true if the piecewise linear function is non-decreasing.
static PiecewiseLinearFunction * CreateEarlyTardyFunction(int64_t reference, int64_t earliness_slope, int64_t tardiness_slope)
void AddConstantToX(int64_t constant)
static PiecewiseLinearFunction * CreatePiecewiseLinearFunction(std::vector< int64_t > points_x, std::vector< int64_t > points_y, std::vector< int64_t > slopes, std::vector< int64_t > other_points_x)
static PiecewiseLinearFunction * CreateEarlyTardyFunctionWithSlack(int64_t early_slack, int64_t late_slack, int64_t earliness_slope, int64_t tardiness_slope)
int64_t Value(int64_t x) const
Returns the value of the piecewise linear function for x.
std::pair< int64_t, int64_t > GetSmallestRangeGreaterThanValue(int64_t range_start, int64_t range_end, int64_t value) const
bool InDomain(int64_t x) const
Returns if x is in the domain of the function.
void Subtract(const PiecewiseLinearFunction &other)
int64_t end_x() const
Returns the end of the segment's domain.
int64_t start_y() const
Returns the value at the start of the segment's domain.
static bool FindComparator(int64_t point, const PiecewiseSegment &segment)
Comparison method useful for finding in which segment a point belongs.
static bool SortComparator(const PiecewiseSegment &segment1, const PiecewiseSegment &segment2)
Comparison method useful for sorting a sequence of segments.
std::string DebugString() const
void ExpandEnd(int64_t end_x)
void AddConstantToY(int64_t constant)
Adds 'constant' to the 'y' the segments.
int64_t slope() const
Returns the segment's slope.
int64_t start_x() const
Returns the start of the segment's domain.
void AddConstantToX(int64_t constant)
Adds 'constant' to the 'x' the segments.
PiecewiseSegment(int64_t point_x, int64_t point_y, int64_t slope, int64_t other_point_x)
PiecewiseSegment.
int64_t Value(int64_t x) const
Returns the value of the segment at point x.
int64_t end_y() const
Returns the value at the end of the segment's domain.
In SWIG mode, we don't want anything besides these top-level includes.
int64_t CapAdd(int64_t x, int64_t y)
int64_t CapSub(int64_t x, int64_t y)
int64_t CapProd(int64_t x, int64_t y)
static const uint64_t kuint64max
static const int64_t kint64max
static const int64_t kint64min