1#![cfg_attr(feature = "generic-simd", feature(portable_simd))]
35
36#![deny(missing_docs)]
37#![cfg_attr(not(feature = "runtime-dispatch-simd"), no_std)]
38
39#[cfg(not(feature = "runtime-dispatch-simd"))]
40use core::mem;
41#[cfg(feature = "runtime-dispatch-simd")]
42use std::mem;
43
44mod naive;
45pub use naive::*;
46mod integer_simd;
47
48#[cfg(any(
49 all(
50 feature = "runtime-dispatch-simd",
51 any(target_arch = "x86", target_arch = "x86_64")
52 ),
53 target_arch = "aarch64",
54 target_arch = "wasm32",
55 feature = "generic-simd"
56))]
57mod simd;
58
59pub fn count(haystack: &[u8], needle: u8) -> usize {
69 if haystack.len() >= 32 {
70 #[cfg(all(feature = "runtime-dispatch-simd", target_arch = "x86_64"))]
71 {
72 if is_x86_feature_detected!("avx2") {
73 unsafe {
74 return simd::x86_avx2::chunk_count(haystack, needle);
75 }
76 }
77 }
78
79 #[cfg(feature = "generic-simd")]
80 return simd::generic::chunk_count(haystack, needle);
81 }
82
83 if haystack.len() >= 16 {
84 #[cfg(all(
85 feature = "runtime-dispatch-simd",
86 any(target_arch = "x86", target_arch = "x86_64"),
87 not(feature = "generic-simd")
88 ))]
89 {
90 if is_x86_feature_detected!("sse2") {
91 unsafe {
92 return simd::x86_sse2::chunk_count(haystack, needle);
93 }
94 }
95 }
96 #[cfg(all(target_arch = "aarch64", not(feature = "generic_simd")))]
97 {
98 unsafe {
99 return simd::aarch64::chunk_count(haystack, needle);
100 }
101 }
102
103 #[cfg(target_arch = "wasm32")]
104 {
105 unsafe {
106 return simd::wasm::chunk_count(haystack, needle);
107 }
108 }
109 }
110
111 if haystack.len() >= mem::size_of::<usize>() {
112 return integer_simd::chunk_count(haystack, needle);
113 }
114
115 naive_count(haystack, needle)
116}
117
118pub fn num_chars(utf8_chars: &[u8]) -> usize {
131 if utf8_chars.len() >= 32 {
132 #[cfg(all(feature = "runtime-dispatch-simd", target_arch = "x86_64"))]
133 {
134 if is_x86_feature_detected!("avx2") {
135 unsafe {
136 return simd::x86_avx2::chunk_num_chars(utf8_chars);
137 }
138 }
139 }
140
141 #[cfg(feature = "generic-simd")]
142 return simd::generic::chunk_num_chars(utf8_chars);
143 }
144
145 if utf8_chars.len() >= 16 {
146 #[cfg(all(
147 feature = "runtime-dispatch-simd",
148 any(target_arch = "x86", target_arch = "x86_64"),
149 not(feature = "generic-simd")
150 ))]
151 {
152 if is_x86_feature_detected!("sse2") {
153 unsafe {
154 return simd::x86_sse2::chunk_num_chars(utf8_chars);
155 }
156 }
157 }
158 #[cfg(all(target_arch = "aarch64", not(feature = "generic_simd")))]
159 {
160 unsafe {
161 return simd::aarch64::chunk_num_chars(utf8_chars);
162 }
163 }
164
165 #[cfg(target_arch = "wasm32")]
166 {
167 unsafe {
168 return simd::wasm::chunk_num_chars(utf8_chars);
169 }
170 }
171 }
172
173 if utf8_chars.len() >= mem::size_of::<usize>() {
174 return integer_simd::chunk_num_chars(utf8_chars);
175 }
176
177 naive_num_chars(utf8_chars)
178}