Coverage Report

Created: 2025-12-31 10:39

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/work/workdir/UnpackedTarball/harfbuzz/src/hb-cache.hh
Line
Count
Source
1
/*
2
 * Copyright © 2012  Google, Inc.
3
 *
4
 *  This is part of HarfBuzz, a text shaping library.
5
 *
6
 * Permission is hereby granted, without written agreement and without
7
 * license or royalty fees, to use, copy, modify, and distribute this
8
 * software and its documentation for any purpose, provided that the
9
 * above copyright notice and the following two paragraphs appear in
10
 * all copies of this software.
11
 *
12
 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13
 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14
 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15
 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16
 * DAMAGE.
17
 *
18
 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20
 * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
21
 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22
 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23
 *
24
 * Google Author(s): Behdad Esfahbod
25
 */
26
27
#ifndef HB_CACHE_HH
28
#define HB_CACHE_HH
29
30
#include "hb.hh"
31
32
33
/* Implements a lockfree and thread-safe cache for int->int functions,
34
 * using (optionally) _relaxed_ atomic integer operations.
35
 *
36
 * The cache is a fixed-size array of 16-bit or 32-bit integers,
37
 * typically 256 elements.
38
 *
39
 * The key is split into two parts: the cache index (high bits)
40
 * and the rest (low bits).
41
 *
42
 * The cache index is used to index into the array.  The array
43
 * member is a 16-bit or 32-bit integer that is used *both*
44
 * to store the low bits of the key, and the value.
45
 *
46
 * The value is stored in the least significant bits of the integer.
47
 * The low bits of the key are stored in the most significant bits
48
 * of the integer.
49
 *
50
 * A cache hit is detected by comparing the low bits of the key
51
 * with the high bits of the integer at the array position indexed
52
 * by the high bits of the key. If they match, the value is extracted
53
 * from the least significant bits of the integer and returned.
54
 * Otherwise, a cache miss is reported.
55
 *
56
 * Cache operations (storage and retrieval) involve just a few
57
 * arithmetic operations and a single memory access.
58
 */
59
60
template <unsigned int key_bits=16,
61
   unsigned int value_bits=8 + 32 - key_bits,
62
   unsigned int cache_bits=8,
63
   bool thread_safe=true>
64
struct hb_cache_t
65
{
66
  using item_t = typename std::conditional<thread_safe,
67
             typename std::conditional<key_bits + value_bits - cache_bits <= 8,
68
                     hb_atomic_t<unsigned char>,
69
                     typename std::conditional<key_bits + value_bits - cache_bits <= 16,
70
                             hb_atomic_t<unsigned short>,
71
                             hb_atomic_t<unsigned int>>::type>::type,
72
             typename std::conditional<key_bits + value_bits - cache_bits <= 8,
73
                     unsigned char,
74
                     typename std::conditional<key_bits + value_bits - cache_bits <= 16,
75
                             unsigned short,
76
                             unsigned int>::type>::type
77
            >::type;
78
79
  static_assert ((key_bits >= cache_bits), "");
80
  static_assert ((key_bits + value_bits <= cache_bits + 8 * sizeof (item_t)), "");
81
82
  static constexpr unsigned MAX_VALUE = (1u << value_bits) - 1;
83
84
1.35k
  hb_cache_t () { clear (); }
hb_cache_t<21u, 3u, 8u, true>::hb_cache_t()
Line
Count
Source
84
61
  hb_cache_t () { clear (); }
hb_cache_t<15u, 8u, 7u, true>::hb_cache_t()
Line
Count
Source
84
61
  hb_cache_t () { clear (); }
hb_cache_t<21u, 19u, 8u, true>::hb_cache_t()
Line
Count
Source
84
61
  hb_cache_t () { clear (); }
Unexecuted instantiation: hb_cache_t<24u, 16u, 8u, true>::hb_cache_t()
hb_cache_t<20u, 20u, 8u, true>::hb_cache_t()
Line
Count
Source
84
1.17k
  hb_cache_t () { clear (); }
85
86
  void clear ()
87
1.54k
  {
88
1.54k
    for (auto &v : values)
89
363k
      v = -1;
90
1.54k
  }
hb_cache_t<21u, 3u, 8u, true>::clear()
Line
Count
Source
87
61
  {
88
61
    for (auto &v : values)
89
15.6k
      v = -1;
90
61
  }
hb_cache_t<15u, 8u, 7u, true>::clear()
Line
Count
Source
87
247
  {
88
247
    for (auto &v : values)
89
31.6k
      v = -1;
90
247
  }
Unexecuted instantiation: hb_cache_t<14u, 1u, 7u, true>::clear()
hb_cache_t<21u, 19u, 8u, true>::clear()
Line
Count
Source
87
61
  {
88
61
    for (auto &v : values)
89
15.6k
      v = -1;
90
61
  }
Unexecuted instantiation: hb_cache_t<24u, 16u, 8u, true>::clear()
hb_cache_t<20u, 20u, 8u, true>::clear()
Line
Count
Source
87
1.17k
  {
88
1.17k
    for (auto &v : values)
89
300k
      v = -1;
90
1.17k
  }
91
92
  HB_HOT
93
  bool get (unsigned int key, unsigned int *value) const
94
1.18G
  {
95
1.18G
    unsigned int k = key & ((1u<<cache_bits)-1);
96
1.18G
    unsigned int v = values[k];
97
1.18G
    if ((key_bits + value_bits - cache_bits == 8 * sizeof (item_t) && (item_t) v == (item_t) -1) ||
98
1.18G
  (v >> value_bits) != (key >> cache_bits))
99
69.4M
      return false;
100
1.12G
    *value = v & ((1u<<value_bits)-1);
101
1.12G
    return true;
102
1.18G
  }
hb_cache_t<15u, 8u, 7u, true>::get(unsigned int, unsigned int*) const
Line
Count
Source
94
185M
  {
95
185M
    unsigned int k = key & ((1u<<cache_bits)-1);
96
185M
    unsigned int v = values[k];
97
185M
    if ((key_bits + value_bits - cache_bits == 8 * sizeof (item_t) && (item_t) v == (item_t) -1) ||
98
185M
  (v >> value_bits) != (key >> cache_bits))
99
87.2k
      return false;
100
185M
    *value = v & ((1u<<value_bits)-1);
101
185M
    return true;
102
185M
  }
Unexecuted instantiation: hb_cache_t<14u, 1u, 7u, true>::get(unsigned int, unsigned int*) const
hb_cache_t<21u, 3u, 8u, true>::get(unsigned int, unsigned int*) const
Line
Count
Source
94
486M
  {
95
486M
    unsigned int k = key & ((1u<<cache_bits)-1);
96
486M
    unsigned int v = values[k];
97
486M
    if ((key_bits + value_bits - cache_bits == 8 * sizeof (item_t) && (item_t) v == (item_t) -1) ||
98
486M
  (v >> value_bits) != (key >> cache_bits))
99
1.71M
      return false;
100
484M
    *value = v & ((1u<<value_bits)-1);
101
484M
    return true;
102
486M
  }
hb_cache_t<21u, 19u, 8u, true>::get(unsigned int, unsigned int*) const
Line
Count
Source
94
514M
  {
95
514M
    unsigned int k = key & ((1u<<cache_bits)-1);
96
514M
    unsigned int v = values[k];
97
514M
    if ((key_bits + value_bits - cache_bits == 8 * sizeof (item_t) && (item_t) v == (item_t) -1) ||
98
506M
  (v >> value_bits) != (key >> cache_bits))
99
67.6M
      return false;
100
446M
    *value = v & ((1u<<value_bits)-1);
101
446M
    return true;
102
514M
  }
Unexecuted instantiation: hb_cache_t<24u, 16u, 8u, true>::get(unsigned int, unsigned int*) const
hb_cache_t<20u, 20u, 8u, true>::get(unsigned int, unsigned int*) const
Line
Count
Source
94
2.76M
  {
95
2.76M
    unsigned int k = key & ((1u<<cache_bits)-1);
96
2.76M
    unsigned int v = values[k];
97
2.76M
    if ((key_bits + value_bits - cache_bits == 8 * sizeof (item_t) && (item_t) v == (item_t) -1) ||
98
2.76M
  (v >> value_bits) != (key >> cache_bits))
99
3.53k
      return false;
100
2.76M
    *value = v & ((1u<<value_bits)-1);
101
2.76M
    return true;
102
2.76M
  }
103
104
  HB_HOT
105
  void set (unsigned int key, unsigned int value)
106
3.26M
  {
107
3.26M
    if (unlikely ((key >> key_bits) || (value >> value_bits)))
108
407k
      return; /* Overflows */
109
2.86M
    set_unchecked (key, value);
110
2.86M
  }
Unexecuted instantiation: hb_cache_t<15u, 8u, 7u, true>::set(unsigned int, unsigned int)
hb_cache_t<21u, 3u, 8u, true>::set(unsigned int, unsigned int)
Line
Count
Source
106
1.71M
  {
107
1.71M
    if (unlikely ((key >> key_bits) || (value >> value_bits)))
108
407k
      return; /* Overflows */
109
1.31M
    set_unchecked (key, value);
110
1.31M
  }
hb_cache_t<21u, 19u, 8u, true>::set(unsigned int, unsigned int)
Line
Count
Source
106
1.54M
  {
107
1.54M
    if (unlikely ((key >> key_bits) || (value >> value_bits)))
108
0
      return; /* Overflows */
109
1.54M
    set_unchecked (key, value);
110
1.54M
  }
Unexecuted instantiation: hb_cache_t<24u, 16u, 8u, true>::set(unsigned int, unsigned int)
hb_cache_t<20u, 20u, 8u, true>::set(unsigned int, unsigned int)
Line
Count
Source
106
3.53k
  {
107
3.53k
    if (unlikely ((key >> key_bits) || (value >> value_bits)))
108
0
      return; /* Overflows */
109
3.53k
    set_unchecked (key, value);
110
3.53k
  }
111
112
  HB_HOT
113
  void set_unchecked (unsigned int key, unsigned int value)
114
2.94M
  {
115
2.94M
    unsigned int k = key & ((1u<<cache_bits)-1);
116
2.94M
    unsigned int v = ((key>>cache_bits)<<value_bits) | value;
117
2.94M
    values[k] = v;
118
2.94M
  }
hb_cache_t<15u, 8u, 7u, true>::set_unchecked(unsigned int, unsigned int)
Line
Count
Source
114
87.2k
  {
115
87.2k
    unsigned int k = key & ((1u<<cache_bits)-1);
116
87.2k
    unsigned int v = ((key>>cache_bits)<<value_bits) | value;
117
87.2k
    values[k] = v;
118
87.2k
  }
Unexecuted instantiation: hb_cache_t<14u, 1u, 7u, true>::set_unchecked(unsigned int, unsigned int)
hb_cache_t<21u, 3u, 8u, true>::set_unchecked(unsigned int, unsigned int)
Line
Count
Source
114
1.31M
  {
115
1.31M
    unsigned int k = key & ((1u<<cache_bits)-1);
116
1.31M
    unsigned int v = ((key>>cache_bits)<<value_bits) | value;
117
1.31M
    values[k] = v;
118
1.31M
  }
hb_cache_t<21u, 19u, 8u, true>::set_unchecked(unsigned int, unsigned int)
Line
Count
Source
114
1.54M
  {
115
1.54M
    unsigned int k = key & ((1u<<cache_bits)-1);
116
1.54M
    unsigned int v = ((key>>cache_bits)<<value_bits) | value;
117
1.54M
    values[k] = v;
118
1.54M
  }
Unexecuted instantiation: hb_cache_t<24u, 16u, 8u, true>::set_unchecked(unsigned int, unsigned int)
hb_cache_t<20u, 20u, 8u, true>::set_unchecked(unsigned int, unsigned int)
Line
Count
Source
114
3.53k
  {
115
3.53k
    unsigned int k = key & ((1u<<cache_bits)-1);
116
3.53k
    unsigned int v = ((key>>cache_bits)<<value_bits) | value;
117
3.53k
    values[k] = v;
118
3.53k
  }
119
120
  private:
121
  item_t values[1u<<cache_bits];
122
};
123
124
125
#endif /* HB_CACHE_HH */