// -*- rust -*- use std; import std.map; import std.util; fn test_simple() { log "*** starting test_simple"; fn eq(&uint x, &uint y) -> bool { ret x == y; } fn hash(&uint u) -> uint { // FIXME: can't use std.util.id since we'd be capturing a type param, // and presently we can't close items over type params. ret u; } let map.hashfn[uint] hasher = hash; let map.eqfn[uint] eqer = eq; let map.hashmap[uint, uint] hm = map.mk_hashmap[uint, uint](hasher, eqer); check (hm.insert(10u, 12u)); check (hm.insert(11u, 13u)); check (hm.insert(12u, 14u)); check (hm.get(11u) == 13u); check (hm.get(12u) == 14u); check (hm.get(10u) == 12u); check (!hm.insert(12u, 14u)); check (hm.get(12u) == 14u); check (!hm.insert(12u, 12u)); check (hm.get(12u) == 12u); log "*** finished test_simple"; } /** * Force map growth and rehashing. */ fn test_growth() { log "*** starting test_growth"; let uint num_to_insert = 64u; fn eq(&uint x, &uint y) -> bool { ret x == y; } fn hash(&uint u) -> uint { // FIXME: can't use std.util.id since we'd be capturing a type param, // and presently we can't close items over type params. ret u; } let map.hashfn[uint] hasher = hash; let map.eqfn[uint] eqer = eq; let map.hashmap[uint, uint] hm = map.mk_hashmap[uint, uint](hasher, eqer); let uint i = 0u; while (i < num_to_insert) { check (hm.insert(i, i * i)); log "inserting " + std._uint.to_str(i, 10u) + " -> " + std._uint.to_str(i * i, 10u); i += 1u; } log "-----"; i = 0u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 1u; } check (hm.insert(num_to_insert, 17u)); check (hm.get(num_to_insert) == 17u); log "-----"; hm.rehash(); i = 0u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 1u; } log "*** finished test_growth"; } fn test_removal() { log "*** starting test_removal"; let uint num_to_insert = 64u; fn eq(&uint x, &uint y) -> bool { ret x == y; } fn hash(&uint u) -> uint { // This hash function intentionally causes collisions between // consecutive integer pairs. ret (u / 2u) * 2u; } check (hash(0u) == hash(1u)); check (hash(2u) == hash(3u)); check (hash(0u) != hash(2u)); let map.hashfn[uint] hasher = hash; let map.eqfn[uint] eqer = eq; let map.hashmap[uint, uint] hm = map.mk_hashmap[uint, uint](hasher, eqer); let uint i = 0u; while (i < num_to_insert) { check (hm.insert(i, i * i)); log "inserting " + std._uint.to_str(i, 10u) + " -> " + std._uint.to_str(i * i, 10u); i += 1u; } check (hm.size() == num_to_insert); log "-----"; log "removing evens"; i = 0u; while (i < num_to_insert) { /** * FIXME (issue #150): we want to check the removed value as in the * following: let util.option[uint] v = hm.remove(i); alt (v) { case (util.some[uint](u)) { check (u == (i * i)); } case (util.none[uint]()) { fail; } } * but we util.option is a tag type so util.some and util.none are * off limits until we parse the dwarf for tag types. */ hm.remove(i); i += 2u; } check (hm.size() == (num_to_insert / 2u)); log "-----"; i = 1u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 2u; } log "-----"; log "rehashing"; hm.rehash(); log "-----"; i = 1u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 2u; } log "-----"; i = 0u; while (i < num_to_insert) { check (hm.insert(i, i * i)); log "inserting " + std._uint.to_str(i, 10u) + " -> " + std._uint.to_str(i * i, 10u); i += 2u; } check (hm.size() == num_to_insert); log "-----"; i = 0u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 1u; } log "-----"; log "rehashing"; hm.rehash(); log "-----"; check (hm.size() == num_to_insert); i = 0u; while (i < num_to_insert) { log "get(" + std._uint.to_str(i, 10u) + ") = " + std._uint.to_str(hm.get(i), 10u); check (hm.get(i) == i * i); i += 1u; } log "*** finished test_removal"; } fn main() { test_simple(); test_growth(); test_removal(); }