harfbuzz/src/test-iter.cc

284 lines
7.4 KiB
C++
Raw Normal View History

2018-12-21 06:23:34 +01:00
/*
* Copyright © 2018 Google, Inc.
*
* This is part of HarfBuzz, a text shaping library.
*
* Permission is hereby granted, without written agreement and without
* license or royalty fees, to use, copy, modify, and distribute this
* software and its documentation for any purpose, provided that the
* above copyright notice and the following two paragraphs appear in
* all copies of this software.
*
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
*
* Google Author(s): Behdad Esfahbod
*/
2018-12-21 23:35:58 +01:00
#include "hb.hh"
2018-12-21 06:23:34 +01:00
#include "hb-iter.hh"
2018-12-21 07:53:27 +01:00
#include "hb-array.hh"
#include "hb-set.hh"
2018-12-27 22:55:18 +01:00
#include "hb-ot-layout-common.hh"
2018-12-21 07:53:27 +01:00
template <typename T>
struct array_iter_t : hb_iter_with_fallback_t<array_iter_t<T>, T&>
2018-12-21 07:53:27 +01:00
{
array_iter_t (hb_array_t<T> arr_) : arr (arr_) {}
typedef T& __item_t__;
2019-01-25 15:34:03 +01:00
static constexpr bool is_random_access_iterator = true;
2018-12-21 07:53:27 +01:00
T& __item_at__ (unsigned i) const { return arr[i]; }
void __forward__ (unsigned n) { arr += n; }
void __rewind__ (unsigned n) { arr -= n; }
2018-12-22 00:46:51 +01:00
unsigned __len__ () const { return arr.length; }
bool operator != (const array_iter_t& o) { return arr != o.arr; }
2018-12-21 07:53:27 +01:00
private:
hb_array_t<T> arr;
};
2018-12-21 09:03:46 +01:00
template <typename T>
struct some_array_t
{
some_array_t (hb_array_t<T> arr_) : arr (arr_) {}
typedef array_iter_t<T> iter_t;
array_iter_t<T> iter () { return array_iter_t<T> (arr); }
operator array_iter_t<T> () { return iter (); }
operator hb_iter_t<array_iter_t<T>> () { return iter (); }
2018-12-21 09:03:46 +01:00
private:
hb_array_t<T> arr;
};
2018-12-27 00:54:27 +01:00
template <typename Iter,
hb_requires (hb_is_iterator (Iter))>
static void
test_iterator_non_default_constructable (Iter it)
2018-12-27 00:54:27 +01:00
{
/* Iterate over a copy of it. */
for (auto c = it.iter (); c; c++)
*c;
/* Same. */
for (auto c = +it; c; c++)
*c;
/* Range-based for over a copy. */
for (auto _ : +it)
(void) _;
it += it.len ();
if (0)
it = it + 10;
if (0)
it = 10 + it;
2018-12-27 00:54:27 +01:00
assert (*it == it[0]);
static_assert (true || it.is_random_access_iterator, "");
static_assert (true || it.is_sorted_iterator, "");
2018-12-27 00:54:27 +01:00
}
template <typename Iter,
hb_requires (hb_is_iterator (Iter))>
static void
test_iterator (Iter it)
{
Iter default_constructed;
assert (!default_constructed);
test_iterator_non_default_constructable (it);
}
template <typename Iterable,
hb_requires (hb_is_iterable (Iterable))>
static void
2018-12-27 22:55:18 +01:00
test_iterable (const Iterable &lst = Null(Iterable))
2018-12-27 00:54:27 +01:00
{
for (auto _ : lst)
(void) _;
2018-12-27 00:54:27 +01:00
// Test that can take iterator from.
test_iterator (lst.iter ());
}
2018-12-21 06:23:34 +01:00
int
main (int argc, char **argv)
{
2018-12-21 06:54:55 +01:00
const int src[10] = {};
int dst[20];
2018-12-21 07:59:37 +01:00
hb_vector_t<int> v;
2018-12-21 06:54:55 +01:00
2018-12-21 07:59:37 +01:00
array_iter_t<const int> s (src); /* Implicit conversion from static array. */
array_iter_t<const int> s2 (v); /* Implicit conversion from vector. */
2018-12-21 07:53:27 +01:00
array_iter_t<int> t (dst);
2018-12-21 06:54:55 +01:00
static_assert (array_iter_t<int>::is_random_access_iterator, "");
2018-12-21 09:03:46 +01:00
some_array_t<const int> a (src);
2018-12-21 06:54:55 +01:00
s2 = s;
2019-01-27 01:03:56 +01:00
hb_iter (src);
hb_iter (src, 2);
2018-12-21 08:47:04 +01:00
hb_fill (t, 42);
hb_copy (s, t);
hb_copy (a.iter (), t);
2018-12-21 06:54:55 +01:00
2018-12-27 00:54:27 +01:00
test_iterable (v);
hb_set_t st;
st << 1 << 15 << 43;
2018-12-27 00:54:27 +01:00
test_iterable (st);
hb_sorted_array_t<int> sa;
(void) static_cast<hb_iter_t<hb_sorted_array_t<int>, hb_sorted_array_t<int>::item_t>&> (sa);
(void) static_cast<hb_iter_t<hb_sorted_array_t<int>, hb_sorted_array_t<int>::__item_t__>&> (sa);
(void) static_cast<hb_iter_t<hb_sorted_array_t<int>, int&>&>(sa);
(void) static_cast<hb_iter_t<hb_sorted_array_t<int>>&>(sa);
(void) static_cast<hb_iter_t<hb_array_t<int>, int&>&> (sa);
test_iterable (sa);
2018-12-27 00:54:27 +01:00
test_iterable<hb_array_t<int>> ();
test_iterable<hb_sorted_array_t<const int>> ();
test_iterable<hb_vector_t<float>> ();
2018-12-27 22:55:18 +01:00
test_iterable<hb_set_t> ();
test_iterable<OT::Coverage> ();
2019-01-09 08:57:16 +01:00
test_iterator (hb_zip (st, v));
test_iterator_non_default_constructable (hb_enumerate (st));
test_iterator_non_default_constructable (hb_enumerate (st, -5));
test_iterator_non_default_constructable (hb_enumerate (hb_iter (st)));
test_iterator_non_default_constructable (hb_enumerate (hb_iter (st) + 1));
2019-05-07 09:05:37 +02:00
test_iterator_non_default_constructable (hb_iter (st) | hb_filter ());
test_iterator_non_default_constructable (hb_iter (st) | hb_map (hb_lidentity));
2019-01-09 08:57:16 +01:00
assert (true == hb_all (st));
assert (false == hb_all (st, 42u));
assert (true == hb_any (st));
2019-05-08 18:32:19 +02:00
assert (false == hb_any (st, 14u));
assert (true == hb_any (st, 14u, [] (unsigned _) { return _ - 1u; }));
assert (true == hb_any (st, [] (unsigned _) { return _ == 15u; }));
assert (true == hb_any (st, 15u));
assert (false == hb_none (st));
2019-05-08 18:32:19 +02:00
assert (false == hb_none (st, 15u));
assert (true == hb_none (st, 17u));
2019-02-16 01:55:08 +01:00
hb_array_t<hb_vector_t<int>> pa;
2019-01-08 03:38:49 +01:00
pa->as_array ();
2019-05-08 19:56:09 +02:00
hb_map_t m;
hb_iter (st);
hb_iter (&st);
2019-01-27 01:03:56 +01:00
+ hb_iter (src)
2019-05-08 19:56:09 +02:00
| hb_map (m)
| hb_map (&m)
| hb_filter ()
2019-05-08 19:56:09 +02:00
| hb_filter (st)
| hb_filter (&st)
| hb_filter (hb_bool)
| hb_filter (hb_bool, hb_identity)
| hb_sink (st)
;
2019-05-08 18:48:55 +02:00
+ hb_iter (src)
| hb_sink (hb_array (dst))
;
2019-02-14 20:40:22 +01:00
+ hb_iter (src)
2019-02-14 23:04:05 +01:00
| hb_apply (&st)
;
2019-03-31 10:11:58 +02:00
+ hb_iter (src)
| hb_map ([&] (int i) -> int { return 1; })
| hb_reduce ([&] (int acc, int value) -> int { return acc; }, 2)
2019-03-31 10:11:58 +02:00
;
2019-05-08 22:51:11 +02:00
using map_pair_t = hb_item_type<hb_map_t>;
2019-05-08 21:09:10 +02:00
+ hb_iter (m)
2019-05-08 23:46:55 +02:00
| hb_map ([] (map_pair_t p) { return p.first * p.second; })
;
2019-05-08 21:09:10 +02:00
2019-05-08 23:46:55 +02:00
m.keys ();
using map_key_t = decltype (*m.keys());
+ hb_iter (m.keys ())
| hb_filter ([] (map_key_t k) { return k < 42; })
2019-05-09 00:08:10 +02:00
| hb_drain
;
m.values ();
using map_value_t = decltype (*m.values());
+ hb_iter (m.values ())
| hb_filter ([] (map_value_t k) { return k < 42; })
| hb_drain
2019-05-08 23:46:55 +02:00
;
2019-05-08 21:47:18 +02:00
unsigned int temp1 = 10;
unsigned int temp2 = 0;
hb_map_t *result =
+ hb_iter (src)
| hb_map ([&] (int i) -> hb_set_t *
{
hb_set_t *set = hb_set_create ();
for (unsigned int i = 0; i < temp1; ++i)
hb_set_add (set, i);
temp1++;
return set;
})
| hb_reduce ([&] (hb_map_t *acc, hb_set_t *value) -> hb_map_t *
{
hb_map_set (acc, temp2++, hb_set_get_population (value));
2019-04-02 17:57:00 +02:00
/* This is not a memory managed language, take care! */
hb_set_destroy (value);
return acc;
}, hb_map_create ())
;
2019-04-02 17:57:00 +02:00
/* The result should be something like 0->10, 1->11, ..., 9->19 */
assert (hb_map_get (result, 9) == 19);
2019-04-22 23:45:23 +02:00
unsigned int temp3 = 0;
+ hb_iter(src)
| hb_map([&] (int i) -> int { return ++temp3; })
| hb_reduce([&] (float acc, int value) -> float { return acc + value; }, 0)
;
hb_map_destroy (result);
2019-02-14 23:04:05 +01:00
+ hb_iter (src)
| hb_drain
;
2019-02-14 20:40:22 +01:00
2019-01-28 22:23:12 +01:00
t << 1;
long vl;
s >> vl;
2019-05-08 19:35:02 +02:00
hb_iota ();
hb_iota (3);
hb_iota (3, 2);
hb_range ();
assert (hb_range (9).len () == 9);
assert (hb_range (2, 9).len () == 7);
assert (hb_range (2, 9, 3).len () == 3);
assert (hb_range (2, 8, 3).len () == 2);
assert (hb_range (2, 7, 3).len () == 2);
assert (hb_range (-2, -9, -3).len () == 3);
assert (hb_range (-2, -8, -3).len () == 2);
assert (hb_range (-2, -7, -3).len () == 2);
2019-05-08 19:35:02 +02:00
2018-12-21 06:23:34 +01:00
return 0;
}