22#include "absl/log/check.h"
23#include "absl/strings/str_cat.h"
24#include "absl/types/span.h"
38enum class RectangleRelationship {
39 TOUCHING_NEITHER_LEFT_OR_BOTTOM,
45RectangleRelationship GetRectangleRelationship(
const Rectangle& rectangle,
49 return RectangleRelationship::OVERLAP;
54 return RectangleRelationship::TOUCHING_LEFT;
58 return RectangleRelationship::TOUCHING_BOTTOM;
60 return RectangleRelationship::TOUCHING_NEITHER_LEFT_OR_BOTTOM;
63bool ShouldPlaceItemAtPosition(
65 std::pair<IntegerValue, IntegerValue> bounding_box_size,
66 absl::Span<const Rectangle> item_positions,
69 if (item_position.
x_max > bounding_box_size.first ||
70 item_position.
y_max > bounding_box_size.second) {
75 if (
i == 0 && (2 * item_position.
x_min >
76 bounding_box_size.first - item_position.
SizeX() ||
77 2 * item_position.
y_min >
78 bounding_box_size.second - item_position.
SizeY())) {
83 bool touches_something_on_left = item_position.
x_min == 0;
84 bool touches_something_on_bottom = item_position.
y_min == 0;
85 for (
const int j : placed_item_indexes) {
87 const RectangleRelationship pos =
88 GetRectangleRelationship(item_position, item_positions[j]);
89 if (pos == RectangleRelationship::OVERLAP) {
92 touches_something_on_left = touches_something_on_left ||
93 pos == RectangleRelationship::TOUCHING_LEFT;
94 touches_something_on_bottom = touches_something_on_bottom ||
95 pos == RectangleRelationship::TOUCHING_BOTTOM;
99 if (!touches_something_on_left || !touches_something_on_bottom) {
105struct PotentialPositionForItem {
108 bool already_explored;
110 Rectangle GetRectangle(IntegerValue x_size, IntegerValue y_size)
const {
111 return {.x_min = x, .x_max = x + x_size, .y_min = y, .y_max = y + y_size};
139bool BruteForceOrthogonalPackingImpl(
140 absl::Span<const IntegerValue> sizes_x,
141 absl::Span<const IntegerValue> sizes_y,
142 std::pair<IntegerValue, IntegerValue> bounding_box_size,
143 IntegerValue smallest_x, IntegerValue smallest_y,
144 absl::Span<Rectangle> item_positions,
Bitset64<int>& placed_item_indexes,
146 potential_item_positions,
147 IntegerValue slack) {
148 const auto add_position_if_valid =
149 [&item_positions, bounding_box_size, &sizes_x, &sizes_y,
150 &placed_item_indexes](
152 IntegerValue x, IntegerValue y) {
154 .x_max = x + sizes_x[
i],
156 .y_max = y + sizes_y[
i]};
157 if (ShouldPlaceItemAtPosition(
i, rect, bounding_box_size,
158 item_positions, placed_item_indexes)) {
159 positions.AppendToLastVector({x, y,
false});
163 const int num_items = sizes_x.size();
165 new_potential_positions.
reserve(num_items,
167 bool has_unplaced_item =
false;
168 for (
int i = 0;
i < num_items; ++
i) {
169 if (placed_item_indexes[
i]) {
172 if (potential_item_positions[
i].empty()) {
176 has_unplaced_item =
true;
177 placed_item_indexes.
Set(
i);
178 for (
const PotentialPositionForItem& potential_position :
179 potential_item_positions[
i]) {
180 if (potential_position.already_explored) {
185 potential_position.GetRectangle(sizes_x[
i], sizes_y[
i]);
186 const Rectangle& item_position = item_positions[
i];
188 IntegerValue slack_loss = 0;
189 if (bounding_box_size.first - item_position.
x_max < smallest_x) {
193 slack_loss += item_position.
SizeY() *
194 (bounding_box_size.first - item_position.
x_max);
196 if (bounding_box_size.second - item_position.
y_max < smallest_y) {
198 slack_loss += item_position.
SizeX() *
199 (bounding_box_size.second - item_position.
y_max);
201 if (slack < slack_loss) {
266 new_potential_positions.
clear();
267 bool is_unfeasible =
false;
268 for (
int j = 0; j < num_items; ++j) {
270 new_potential_positions.
Add({});
273 if (placed_item_indexes[j]) {
278 for (
const int k : placed_item_indexes) {
284 add_position_if_valid(new_potential_positions, j, item_position.
x_max,
285 item_positions[k].y_max);
286 add_position_if_valid(new_potential_positions, j,
287 item_positions[k].x_max, item_position.
y_max);
291 for (
const PotentialPositionForItem& original_position :
292 potential_item_positions[j]) {
293 if (!original_position.GetRectangle(sizes_x[j], sizes_y[j])
294 .IsDisjoint(item_position)) {
305 PotentialPositionForItem position = original_position;
306 position.already_explored =
true;
312 add_position_if_valid(new_potential_positions, j,
313 item_positions[
i].x_max, 0);
314 add_position_if_valid(new_potential_positions, j, 0,
315 item_positions[
i].y_max);
316 if (new_potential_positions[j].empty()) {
319 is_unfeasible =
true;
326 if (BruteForceOrthogonalPackingImpl(
327 sizes_x, sizes_y, bounding_box_size, smallest_x, smallest_y,
328 item_positions, placed_item_indexes, new_potential_positions,
329 slack - slack_loss)) {
335 placed_item_indexes.
Set(
i,
false);
337 return !has_unplaced_item;
340bool BruteForceOrthogonalPackingNoPreprocessing(
341 const absl::Span<PermutableItem> items,
342 const std::pair<IntegerValue, IntegerValue> bounding_box_size) {
343 IntegerValue smallest_x = std::numeric_limits<IntegerValue>::max();
344 IntegerValue smallest_y = std::numeric_limits<IntegerValue>::max();
345 const int num_items = items.size();
347 IntegerValue slack = bounding_box_size.first * bounding_box_size.second;
350 smallest_x = std::min(smallest_x, item.size_x);
351 smallest_y = std::min(smallest_y, item.size_y);
352 slack -= item.size_x * item.size_y;
353 if (item.size_x > bounding_box_size.first ||
354 item.size_y > bounding_box_size.second) {
362 std::sort(items.begin(), items.end(),
364 return std::make_tuple(a.size_x * a.size_y, a.index) >
365 std::make_tuple(b.size_x * b.size_y, b.index);
368 std::vector<IntegerValue> new_sizes_x(num_items);
369 std::vector<IntegerValue> new_sizes_y(num_items);
371 potential_item_positions.
reserve(num_items, num_items);
372 for (
int i = 0;
i < num_items; ++
i) {
373 new_sizes_x[
i] = items[
i].size_x;
374 new_sizes_y[
i] = items[
i].size_y;
375 potential_item_positions.
Add({PotentialPositionForItem{0, 0,
false}});
377 std::vector<Rectangle> item_positions(num_items);
379 const bool found_solution = BruteForceOrthogonalPackingImpl(
380 new_sizes_x, new_sizes_y, bounding_box_size, smallest_x, smallest_y,
381 absl::MakeSpan(item_positions), placed_item_indexes,
382 potential_item_positions, slack);
383 if (!found_solution) {
386 for (
int i = 0;
i < num_items; ++
i) {
387 items[
i].position = item_positions[
i];
459bool PreprocessLargeWithSmallX(
460 absl::Span<PermutableItem>& items,
461 std::pair<IntegerValue, IntegerValue>& bounding_box_size,
462 int max_complexity) {
470 items.begin(), items.end(),
472 const bool a_is_small = 2 * a.size_x <= bounding_box_size.first;
473 const bool b_is_small = 2 * b.size_x <= bounding_box_size.first;
474 const IntegerValue a_size_for_comp =
475 a_is_small ? a.size_x : bounding_box_size.first - a.size_x;
476 const IntegerValue b_size_for_comp =
477 b_is_small ? b.size_x : bounding_box_size.first - b.size_x;
478 return std::make_tuple(a_size_for_comp, !a_is_small, a.index) >
479 std::make_tuple(b_size_for_comp, !b_is_small, b.index);
481 IntegerValue total_large_item_width = 0;
482 IntegerValue largest_small_item_height = 0;
483 for (
int i = items.size() - 1;
i >= 1; --
i) {
484 if (2 * items[
i].size_x <= bounding_box_size.first) {
485 largest_small_item_height = items[
i].size_x;
488 DCHECK_LE(items[
i].size_x + largest_small_item_height,
489 bounding_box_size.first);
493 if (items.subspan(
i).size() > max_complexity) {
496 total_large_item_width += items[
i].size_y;
497 if (BruteForceOrthogonalPackingNoPreprocessing(
499 {bounding_box_size.first, total_large_item_width})) {
500 bounding_box_size.second -= total_large_item_width;
501 for (
auto& item : items.subspan(
i)) {
502 item.position.y_min += bounding_box_size.second;
503 item.position.y_max += bounding_box_size.second;
505 items = items.subspan(0,
i);
514bool PreprocessLargeWithSmallY(
515 absl::Span<PermutableItem>& items,
516 std::pair<IntegerValue, IntegerValue>& bounding_box_size,
517 int max_complexity) {
518 absl::Span<PermutableItem> orig_items = items;
520 std::swap(item.size_x, item.size_y);
521 std::swap(item.position.x_min, item.position.y_min);
522 std::swap(item.position.x_max, item.position.y_max);
524 std::swap(bounding_box_size.first, bounding_box_size.second);
526 PreprocessLargeWithSmallX(items, bounding_box_size, max_complexity);
527 std::swap(bounding_box_size.first, bounding_box_size.second);
529 std::swap(item.size_x, item.size_y);
530 std::swap(item.position.x_min, item.position.y_min);
531 std::swap(item.position.x_max, item.position.y_max);
547 std::pair<IntegerValue, IntegerValue>& bounding_box_size,
548 int max_complexity) {
549 const int num_items = items.size();
550 if (num_items == 1) {
553 IntegerValue smallest_x = std::numeric_limits<IntegerValue>::max();
554 IntegerValue largest_x = std::numeric_limits<IntegerValue>::min();
555 IntegerValue smallest_y = std::numeric_limits<IntegerValue>::max();
556 IntegerValue largest_y = std::numeric_limits<IntegerValue>::min();
557 int largest_x_idx = -1;
558 int largest_y_idx = -1;
559 for (
int i = 0;
i < num_items; ++
i) {
560 if (items[
i].size_x > largest_x) {
561 largest_x = items[
i].size_x;
564 if (items[
i].size_y > largest_y) {
565 largest_y = items[
i].size_y;
568 smallest_x = std::min(smallest_x, items[
i].size_x);
569 smallest_y = std::min(smallest_y, items[
i].size_y);
572 if (largest_x > bounding_box_size.first ||
573 largest_y > bounding_box_size.second) {
577 const auto remove_item = [&items](
int index_to_remove) {
578 std::swap(items[index_to_remove], items.back());
579 items.remove_suffix(1);
581 if (largest_x + smallest_x > bounding_box_size.first) {
584 bounding_box_size.second -= items[largest_x_idx].size_y;
585 items[largest_x_idx].position = {
588 .y_min = bounding_box_size.second,
589 .y_max = bounding_box_size.second + items[largest_x_idx].size_y};
590 remove_item(largest_x_idx);
591 Preprocess(items, bounding_box_size, max_complexity);
594 if (largest_y + smallest_y > bounding_box_size.second) {
595 bounding_box_size.first -= items[largest_y_idx].size_x;
596 items[largest_y_idx].position = {
597 .x_min = bounding_box_size.first,
598 .x_max = bounding_box_size.first + items[largest_y_idx].size_x,
601 remove_item(largest_y_idx);
602 Preprocess(items, bounding_box_size, max_complexity);
606 if (PreprocessLargeWithSmallX(items, bounding_box_size, max_complexity)) {
607 Preprocess(items, bounding_box_size, max_complexity);
611 if (PreprocessLargeWithSmallY(items, bounding_box_size, max_complexity)) {
612 Preprocess(items, bounding_box_size, max_complexity);
620 absl::Span<const IntegerValue> sizes_x,
621 absl::Span<const IntegerValue> sizes_y,
622 std::pair<IntegerValue, IntegerValue> bounding_box_size,
623 int max_complexity) {
624 const int num_items = sizes_x.size();
626 if (num_items > 2 * max_complexity) {
633 std::vector<PermutableItem> items(num_items);
634 for (
int i = 0;
i < num_items; ++
i) {
636 .size_x = sizes_x[
i], .size_y = sizes_y[
i], .index =
i, .position = {}};
638 absl::Span<PermutableItem> post_processed_items = absl::MakeSpan(items);
639 std::pair<IntegerValue, IntegerValue> post_processed_bounding_box_size =
641 const bool post_processed =
642 Preprocess(post_processed_items, post_processed_bounding_box_size,
644 if (post_processed_items.size() > max_complexity) {
647 const bool is_feasible = BruteForceOrthogonalPackingNoPreprocessing(
648 post_processed_items, post_processed_bounding_box_size);
649 VLOG_EVERY_N_SEC(2, 3)
650 <<
"Solved by brute force a problem of " << num_items <<
" items"
651 << (post_processed ? absl::StrCat(
" (", post_processed_items.size(),
652 " after preprocessing)")
654 <<
": solution " << (is_feasible ?
"found" :
"not found") <<
".";
658 std::vector<Rectangle> result(num_items);
660 result[item.index] = item.position;
662 VLOG_EVERY_N_SEC(3, 3) <<
"Found a feasible packing by brute force. Dot:\n "
665 .x_max = bounding_box_size.first,
667 .y_max = bounding_box_size.second},
670 .positions_for_solution = result};
void Set(IndexType i)
Sets the bit at position i to 1.
void clear()
Restore to empty vector<vector<>>.
size_t num_entries() const
void reserve(int size)
Reserve memory if it is already known or tightly estimated.
void AppendToLastVector(const V &value)
int Add(absl::Span< const V > values)
BruteForceResult BruteForceOrthogonalPacking(absl::Span< const IntegerValue > sizes_x, absl::Span< const IntegerValue > sizes_y, std::pair< IntegerValue, IntegerValue > bounding_box_size, int max_complexity)
bool Preprocess(absl::Span< PermutableItem > &items, std::pair< IntegerValue, IntegerValue > &bounding_box_size, int max_complexity)
Exposed for testing.
static constexpr int kMaxProblemSize
std::string RenderDot(std::optional< Rectangle > bb, absl::Span< const Rectangle > solution, std::string_view extra_dot_payload)
In SWIG mode, we don't want anything besides these top-level includes.
IntegerValue SizeY() const
IntegerValue SizeX() const