2014-06-05 09:55:49 +02:00
|
|
|
// The Computer Language Benchmarks Game
|
|
|
|
// http://benchmarksgame.alioth.debian.org/
|
2013-11-14 09:11:33 +01:00
|
|
|
//
|
2014-06-05 09:55:49 +02:00
|
|
|
// contributed by the Rust Project Developers
|
|
|
|
|
|
|
|
// Copyright (c) 2013-2014 The Rust Project Developers
|
|
|
|
//
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions
|
|
|
|
// are met:
|
|
|
|
//
|
|
|
|
// - Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
//
|
|
|
|
// - Redistributions in binary form must reproduce the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer in
|
|
|
|
// the documentation and/or other materials provided with the
|
|
|
|
// distribution.
|
|
|
|
//
|
|
|
|
// - Neither the name of "The Computer Language Benchmarks Game" nor
|
|
|
|
// the name of "The Computer Language Shootout Benchmarks" nor the
|
|
|
|
// names of its contributors may be used to endorse or promote
|
|
|
|
// products derived from this software without specific prior
|
|
|
|
// written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
// COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
// HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
|
|
// OF THE POSSIBILITY OF SUCH DAMAGE.
|
2013-11-14 09:11:33 +01:00
|
|
|
|
2014-06-25 19:41:16 -07:00
|
|
|
// no-pretty-expanded FIXME #15189
|
2014-06-25 19:15:06 -07:00
|
|
|
|
2014-06-07 11:13:26 -07:00
|
|
|
use std::sync::Arc;
|
2014-05-09 17:32:06 +02:00
|
|
|
|
2013-11-14 09:11:33 +01:00
|
|
|
//
|
|
|
|
// Utilities.
|
|
|
|
//
|
|
|
|
|
|
|
|
// returns an infinite iterator of repeated applications of f to x,
|
|
|
|
// i.e. [x, f(x), f(f(x)), ...], as haskell iterate function.
|
2014-04-02 09:47:11 -07:00
|
|
|
fn iterate<'a, T>(x: T, f: |&T|: 'a -> T) -> Iterate<'a, T> {
|
2013-11-14 23:05:33 +01:00
|
|
|
Iterate {f: f, next: x}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2013-12-09 23:16:18 -08:00
|
|
|
struct Iterate<'a, T> {
|
2014-04-02 09:47:11 -07:00
|
|
|
f: |&T|: 'a -> T,
|
2014-01-24 11:02:03 -08:00
|
|
|
next: T
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, T> Iterator<T> for Iterate<'a, T> {
|
2013-11-14 09:11:33 +01:00
|
|
|
fn next(&mut self) -> Option<T> {
|
|
|
|
let mut res = (self.f)(&self.next);
|
2014-02-01 04:35:36 +08:00
|
|
|
std::mem::swap(&mut res, &mut self.next);
|
2013-11-14 09:11:33 +01:00
|
|
|
Some(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// a linked list using borrowed next.
|
2014-08-27 21:46:52 -04:00
|
|
|
enum List<'a, T:'a> {
|
2013-11-14 09:11:33 +01:00
|
|
|
Nil,
|
2013-12-09 23:16:18 -08:00
|
|
|
Cons(T, &'a List<'a, T>)
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2014-08-27 21:46:52 -04:00
|
|
|
struct ListIterator<'a, T:'a> {
|
2014-01-24 11:02:03 -08:00
|
|
|
cur: &'a List<'a, T>
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, T> List<'a, T> {
|
|
|
|
fn iter(&'a self) -> ListIterator<'a, T> {
|
2013-11-14 09:11:33 +01:00
|
|
|
ListIterator{cur: self}
|
|
|
|
}
|
|
|
|
}
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, T> Iterator<&'a T> for ListIterator<'a, T> {
|
|
|
|
fn next(&mut self) -> Option<&'a T> {
|
2013-11-14 09:11:33 +01:00
|
|
|
match *self.cur {
|
|
|
|
Nil => None,
|
|
|
|
Cons(ref elt, next) => {
|
|
|
|
self.cur = next;
|
|
|
|
Some(elt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// preprocess
|
|
|
|
//
|
|
|
|
|
|
|
|
// Takes a pieces p on the form [(y1, x1), (y2, x2), ...] and returns
|
|
|
|
// every possible transformations (the 6 rotations with their
|
|
|
|
// corresponding mirrored piece), with, as minimum coordinates, (0,
|
|
|
|
// 0). If all is false, only generate half of the possibilities (used
|
2014-08-01 19:42:13 -04:00
|
|
|
// to break the symmetry of the board).
|
2014-03-05 15:28:08 -08:00
|
|
|
fn transform(piece: Vec<(int, int)> , all: bool) -> Vec<Vec<(int, int)>> {
|
|
|
|
let mut res: Vec<Vec<(int, int)>> =
|
2013-11-14 09:11:33 +01:00
|
|
|
// rotations
|
2013-11-14 23:22:44 +01:00
|
|
|
iterate(piece, |rot| rot.iter().map(|&(y, x)| (x + y, -y)).collect())
|
2013-11-14 09:11:33 +01:00
|
|
|
.take(if all {6} else {3})
|
|
|
|
// mirror
|
2013-11-14 23:22:44 +01:00
|
|
|
.flat_map(|cur_piece| {
|
|
|
|
iterate(cur_piece, |mir| mir.iter().map(|&(y, x)| (x, y)).collect())
|
|
|
|
.take(2)
|
2014-03-05 15:28:08 -08:00
|
|
|
}).collect();
|
2013-11-14 09:11:33 +01:00
|
|
|
|
|
|
|
// translating to (0, 0) as minimum coordinates.
|
2014-09-14 20:27:36 -07:00
|
|
|
for cur_piece in res.iter_mut() {
|
2013-11-14 23:22:44 +01:00
|
|
|
let (dy, dx) = *cur_piece.iter().min_by(|e| *e).unwrap();
|
2014-09-14 20:27:36 -07:00
|
|
|
for &(ref mut y, ref mut x) in cur_piece.iter_mut() {
|
2013-11-14 09:11:33 +01:00
|
|
|
*y -= dy; *x -= dx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
2014-08-01 19:42:13 -04:00
|
|
|
// A mask is a piece somewhere on the board. It is represented as a
|
2013-11-14 09:11:33 +01:00
|
|
|
// u64: for i in the first 50 bits, m[i] = 1 if the cell at (i/5, i%5)
|
2014-08-01 19:42:13 -04:00
|
|
|
// is occupied. m[50 + id] = 1 if the identifier of the piece is id.
|
2013-11-14 09:11:33 +01:00
|
|
|
|
|
|
|
// Takes a piece with minimum coordinate (0, 0) (as generated by
|
|
|
|
// transform). Returns the corresponding mask if p translated by (dy,
|
|
|
|
// dx) is on the board.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn mask(dy: int, dx: int, id: uint, p: &Vec<(int, int)>) -> Option<u64> {
|
2013-11-14 09:11:33 +01:00
|
|
|
let mut m = 1 << (50 + id);
|
|
|
|
for &(y, x) in p.iter() {
|
|
|
|
let x = x + dx + (y + (dy % 2)) / 2;
|
|
|
|
if x < 0 || x > 4 {return None;}
|
|
|
|
let y = y + dy;
|
|
|
|
if y < 0 || y > 9 {return None;}
|
2014-04-21 17:58:52 -04:00
|
|
|
m |= 1 << (y * 5 + x) as uint;
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
Some(m)
|
|
|
|
}
|
|
|
|
|
2014-05-09 17:32:06 +02:00
|
|
|
// Makes every possible masks. masks[i][id] correspond to every
|
2013-11-14 09:11:33 +01:00
|
|
|
// possible masks for piece with identifier id with minimum coordinate
|
|
|
|
// (i/5, i%5).
|
2014-03-05 14:02:44 -08:00
|
|
|
fn make_masks() -> Vec<Vec<Vec<u64> > > {
|
|
|
|
let pieces = vec!(
|
2014-04-21 17:58:52 -04:00
|
|
|
vec!((0i,0i),(0,1),(0,2),(0,3),(1,3)),
|
|
|
|
vec!((0i,0i),(0,2),(0,3),(1,0),(1,1)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(1,2),(2,1)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(1,1),(2,1)),
|
|
|
|
vec!((0i,0i),(0,2),(1,0),(1,1),(2,1)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(1,1),(1,2)),
|
|
|
|
vec!((0i,0i),(0,1),(1,1),(1,2),(2,1)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(1,0),(1,2)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(1,2),(1,3)),
|
|
|
|
vec!((0i,0i),(0,1),(0,2),(0,3),(1,2)));
|
2014-05-09 17:32:06 +02:00
|
|
|
|
2014-08-01 19:42:13 -04:00
|
|
|
// To break the central symmetry of the problem, every
|
2014-05-09 17:32:06 +02:00
|
|
|
// transformation must be taken except for one piece (piece 3
|
|
|
|
// here).
|
|
|
|
let transforms: Vec<Vec<Vec<(int, int)>>> =
|
2014-09-14 20:27:36 -07:00
|
|
|
pieces.into_iter().enumerate()
|
2014-05-09 17:32:06 +02:00
|
|
|
.map(|(id, p)| transform(p, id != 3))
|
|
|
|
.collect();
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
range(0i, 50).map(|yx| {
|
2014-05-09 17:32:06 +02:00
|
|
|
transforms.iter().enumerate().map(|(id, t)| {
|
|
|
|
t.iter().filter_map(|p| mask(yx / 5, yx % 5, id, p)).collect()
|
|
|
|
}).collect()
|
|
|
|
}).collect()
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if all coordinates can be covered by an unused piece and that
|
|
|
|
// all unused piece can be placed on the board.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn is_board_unfeasible(board: u64, masks: &Vec<Vec<Vec<u64>>>) -> bool {
|
2013-11-14 09:11:33 +01:00
|
|
|
let mut coverable = board;
|
2014-05-09 17:32:06 +02:00
|
|
|
for (i, masks_at) in masks.iter().enumerate() {
|
|
|
|
if board & 1 << i != 0 { continue; }
|
|
|
|
for (cur_id, pos_masks) in masks_at.iter().enumerate() {
|
|
|
|
if board & 1 << (50 + cur_id) != 0 { continue; }
|
|
|
|
for &cur_m in pos_masks.iter() {
|
|
|
|
if cur_m & board != 0 { continue; }
|
|
|
|
coverable |= cur_m;
|
|
|
|
// if every coordinates can be covered and every
|
|
|
|
// piece can be used.
|
|
|
|
if coverable == (1 << 60) - 1 { return false; }
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
}
|
2014-05-09 17:32:06 +02:00
|
|
|
if coverable & 1 << i == 0 { return true; }
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2014-05-09 17:32:06 +02:00
|
|
|
true
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Filter the masks that we can prove to result to unfeasible board.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn filter_masks(masks: &mut Vec<Vec<Vec<u64>>>) {
|
|
|
|
for i in range(0, masks.len()) {
|
2014-10-11 01:46:59 +02:00
|
|
|
for j in range(0, (*masks)[i].len()) {
|
2014-11-06 12:25:16 -05:00
|
|
|
masks[i][j] =
|
2014-10-11 01:46:59 +02:00
|
|
|
(*masks)[i][j].iter().map(|&m| m)
|
2013-11-14 09:11:33 +01:00
|
|
|
.filter(|&m| !is_board_unfeasible(m, masks))
|
2014-05-09 17:32:06 +02:00
|
|
|
.collect();
|
|
|
|
}
|
|
|
|
}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Gets the identifier of a mask.
|
|
|
|
fn get_id(m: u64) -> u8 {
|
2014-05-09 17:32:06 +02:00
|
|
|
for id in range(0u8, 10) {
|
2014-04-21 17:58:52 -04:00
|
|
|
if m & (1 << (id + 50) as uint) != 0 {return id;}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2014-10-09 15:17:22 -04:00
|
|
|
panic!("{:016x} does not have a valid identifier", m);
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
2014-06-05 09:55:49 +02:00
|
|
|
// Converts a list of mask to a Vec<u8>.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn to_vec(raw_sol: &List<u64>) -> Vec<u8> {
|
|
|
|
let mut sol = Vec::from_elem(50, '.' as u8);
|
2013-11-14 09:11:33 +01:00
|
|
|
for &m in raw_sol.iter() {
|
2014-05-09 17:32:06 +02:00
|
|
|
let id = '0' as u8 + get_id(m);
|
|
|
|
for i in range(0u, 50) {
|
2014-03-05 15:28:08 -08:00
|
|
|
if m & 1 << i != 0 {
|
2014-11-06 12:25:16 -05:00
|
|
|
sol[i] = id;
|
2014-03-05 15:28:08 -08:00
|
|
|
}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
}
|
2014-05-09 17:32:06 +02:00
|
|
|
sol
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
2014-06-05 09:55:49 +02:00
|
|
|
// Prints a solution in Vec<u8> form.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn print_sol(sol: &Vec<u8>) {
|
|
|
|
for (i, c) in sol.iter().enumerate() {
|
2014-01-09 21:06:55 +11:00
|
|
|
if (i) % 5 == 0 { println!(""); }
|
|
|
|
if (i + 5) % 10 == 0 { print!(" "); }
|
2014-05-09 17:32:06 +02:00
|
|
|
print!("{} ", *c as char);
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
2014-01-09 21:06:55 +11:00
|
|
|
println!("");
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// The data managed during the search
|
|
|
|
struct Data {
|
|
|
|
// Number of solution found.
|
|
|
|
nb: int,
|
|
|
|
// Lexicographically minimal solution found.
|
2014-05-09 17:32:06 +02:00
|
|
|
min: Vec<u8>,
|
2013-11-14 09:11:33 +01:00
|
|
|
// Lexicographically maximal solution found.
|
2014-05-09 17:32:06 +02:00
|
|
|
max: Vec<u8>
|
|
|
|
}
|
|
|
|
impl Data {
|
|
|
|
fn new() -> Data {
|
|
|
|
Data {nb: 0, min: vec!(), max: vec!()}
|
|
|
|
}
|
|
|
|
fn reduce_from(&mut self, other: Data) {
|
|
|
|
self.nb += other.nb;
|
|
|
|
let Data { min: min, max: max, ..} = other;
|
|
|
|
if min < self.min { self.min = min; }
|
|
|
|
if max > self.max { self.max = max; }
|
|
|
|
}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Records a new found solution. Returns false if the search must be
|
|
|
|
// stopped.
|
2014-05-09 17:32:06 +02:00
|
|
|
fn handle_sol(raw_sol: &List<u64>, data: &mut Data) {
|
2014-08-01 19:42:13 -04:00
|
|
|
// because we break the symmetry, 2 solutions correspond to a call
|
2013-11-14 09:11:33 +01:00
|
|
|
// to this method: the normal solution, and the same solution in
|
|
|
|
// reverse order, i.e. the board rotated by half a turn.
|
|
|
|
data.nb += 2;
|
2014-05-09 17:32:06 +02:00
|
|
|
let sol1 = to_vec(raw_sol);
|
|
|
|
let sol2: Vec<u8> = sol1.iter().rev().map(|x| *x).collect();
|
2013-11-14 09:11:33 +01:00
|
|
|
|
|
|
|
if data.nb == 2 {
|
|
|
|
data.min = sol1.clone();
|
|
|
|
data.max = sol1.clone();
|
|
|
|
}
|
|
|
|
|
2014-05-09 17:32:06 +02:00
|
|
|
if sol1 < data.min {data.min = sol1;}
|
|
|
|
else if sol1 > data.max {data.max = sol1;}
|
|
|
|
if sol2 < data.min {data.min = sol2;}
|
|
|
|
else if sol2 > data.max {data.max = sol2;}
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn search(
|
2014-05-09 17:32:06 +02:00
|
|
|
masks: &Vec<Vec<Vec<u64>>>,
|
2013-11-14 09:11:33 +01:00
|
|
|
board: u64,
|
2014-05-09 17:32:06 +02:00
|
|
|
mut i: uint,
|
2013-11-14 09:11:33 +01:00
|
|
|
cur: List<u64>,
|
|
|
|
data: &mut Data)
|
|
|
|
{
|
|
|
|
// Search for the lesser empty coordinate.
|
|
|
|
while board & (1 << i) != 0 && i < 50 {i += 1;}
|
|
|
|
// the board is full: a solution is found.
|
|
|
|
if i >= 50 {return handle_sol(&cur, data);}
|
2014-10-11 01:46:59 +02:00
|
|
|
let masks_at = &masks[i];
|
2013-11-14 09:11:33 +01:00
|
|
|
|
|
|
|
// for every unused piece
|
2014-10-31 05:40:15 -04:00
|
|
|
for id in range(0u, 10).filter(|&id| board & (1 << (id + 50)) == 0) {
|
2013-11-14 09:11:33 +01:00
|
|
|
// for each mask that fits on the board
|
2014-10-11 01:46:59 +02:00
|
|
|
for m in masks_at[id].iter().filter(|&m| board & *m == 0) {
|
2014-08-01 19:42:13 -04:00
|
|
|
// This check is too costly.
|
2013-11-14 09:11:33 +01:00
|
|
|
//if is_board_unfeasible(board | m, masks) {continue;}
|
2014-07-21 20:54:28 -07:00
|
|
|
search(masks, board | *m, i + 1, Cons(*m, &cur), data);
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
}
|
2014-05-09 17:32:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn par_search(masks: Vec<Vec<Vec<u64>>>) -> Data {
|
|
|
|
let masks = Arc::new(masks);
|
|
|
|
let (tx, rx) = channel();
|
|
|
|
|
|
|
|
// launching the search in parallel on every masks at minimum
|
|
|
|
// coordinate (0,0)
|
2014-10-11 01:46:59 +02:00
|
|
|
for m in (*masks)[0].iter().flat_map(|masks_pos| masks_pos.iter()) {
|
2014-05-09 17:32:06 +02:00
|
|
|
let masks = masks.clone();
|
|
|
|
let tx = tx.clone();
|
2014-07-21 20:54:28 -07:00
|
|
|
let m = *m;
|
2014-05-09 17:32:06 +02:00
|
|
|
spawn(proc() {
|
|
|
|
let mut data = Data::new();
|
|
|
|
search(&*masks, m, 1, Cons(m, &Nil), &mut data);
|
|
|
|
tx.send(data);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// collecting the results
|
|
|
|
drop(tx);
|
|
|
|
let mut data = rx.recv();
|
|
|
|
for d in rx.iter() { data.reduce_from(d); }
|
|
|
|
data
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn main () {
|
2014-05-09 17:32:06 +02:00
|
|
|
let mut masks = make_masks();
|
|
|
|
filter_masks(&mut masks);
|
|
|
|
let data = par_search(masks);
|
2013-11-14 09:11:33 +01:00
|
|
|
println!("{} solutions found", data.nb);
|
2014-05-09 17:32:06 +02:00
|
|
|
print_sol(&data.min);
|
|
|
|
print_sol(&data.max);
|
2014-01-09 21:06:55 +11:00
|
|
|
println!("");
|
2013-11-14 09:11:33 +01:00
|
|
|
}
|