-
Notifications
You must be signed in to change notification settings - Fork 106
/
Copy patharray_block.mbt
329 lines (296 loc) · 9.17 KB
/
array_block.mbt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
// Copyright 2025 International Digital Economy Academy
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
///|
/// Copies elements from one array to another array. Works correctly even when
/// the source and destination arrays overlap.
///
/// Parameters:
///
/// * `dst` : The destination array where elements will be copied to.
/// * `dst_offset` : The starting index in the destination array where elements
/// should be copied.
/// * `src` : The source array from which elements will be copied.
/// * `src_offset` : The starting index in the source array from which elements
/// should be copied.
/// * `len` : The number of elements to copy.
///
/// The behavior is undefined if any of the following conditions are met:
///
/// * `len` is negative
/// * `dst_offset` is negative
/// * `src_offset` is negative
/// * `dst_offset + len` exceeds the length of destination array
/// * `src_offset + len` exceeds the length of source array
///
/// Example:
///
/// ```moonbit
/// test "Array::unsafe_blit" {
/// let src = [1, 2, 3, 4, 5]
/// let dst = [0, 0, 0, 0, 0]
/// Array::unsafe_blit(dst, 1, src, 2, 2)
/// inspect!(dst, content="[0, 3, 4, 0, 0]")
/// }
/// ```
///
/// @alert unsafe "Panic if the indices or length are out of bounds"
pub fn Array::unsafe_blit[A](
dst : Array[A],
dst_offset : Int,
src : Array[A],
src_offset : Int,
len : Int
) -> Unit {
FixedArray::unsafe_blit(
dst.buffer()._,
dst_offset,
src.buffer()._,
src_offset,
len,
)
}
///|
/// Copies elements from a fixed-size array to a dynamic array. The arrays may
/// overlap, in which case the copy is performed in a way that preserves the
/// data.
///
/// Parameters:
///
/// * `dst` : The destination dynamic array where elements will be copied to.
/// * `dst_offset` : The starting index in the destination array where copying
/// begins.
/// * `src` : The source fixed-size array from which elements will be copied.
/// * `src_offset` : The starting index in the source array from which copying
/// begins.
/// * `len` : The number of elements to copy.
///
/// Example:
///
/// ```moonbit
/// test "Array::unsafe_blit_fixed" {
/// let src = FixedArray::make(5, 0)
/// let dst = Array::make(5, 0)
/// for i in 0..<5 {
/// src[i] = i + 1
/// }
/// Array::unsafe_blit_fixed(dst, 1, src, 0, 2)
/// inspect!(dst, content="[0, 1, 2, 0, 0]")
/// }
/// ```
pub fn Array::unsafe_blit_fixed[A](
dst : Array[A],
dst_offset : Int,
src : FixedArray[A],
src_offset : Int,
len : Int
) -> Unit {
UninitializedArray::unsafe_blit_fixed(
dst.buffer(),
dst_offset,
src,
src_offset,
len,
)
}
///|
/// Copies elements from one array to another array, with support for growing the
/// destination array if needed. The arrays may overlap, in which case the copy
/// is performed in a way that preserves the data.
///
/// Parameters:
///
/// * `source` : The array to copy elements from.
/// * `destination` : The array to copy elements to. Will be automatically grown
/// if needed to accommodate the copied elements.
/// * `length` : The number of elements to copy.
/// * `source_start` : Starting index in the source array. Defaults to 0.
/// * `destination_start` : Starting index in the destination array. Defaults to
/// 0.
///
/// Example:
///
/// ```moonbit
/// test "Array::blit_to/basic" {
/// let src = [1, 2, 3, 4, 5]
/// let dst = [0, 0]
/// src.blit_to(dst, len=3, dst_offset=1)
/// inspect!(dst, content="[0, 1, 2, 3]")
/// }
/// ```
///
/// Panics if:
///
/// * `len` is negative
/// * `src_offset` is negative
/// * `dst_offset` is negative
/// * `dst_offset + len` exceeds the length of destination array
/// * `src_offset + len` exceeds the length of source array
pub fn Array::blit_to[A](
self : Array[A],
dst : Array[A],
len~ : Int,
src_offset~ : Int = 0,
dst_offset~ : Int = 0
) -> Unit {
guard len >= 0 &&
dst_offset >= 0 &&
src_offset >= 0 &&
dst_offset <= dst.length() &&
src_offset + len <= self.length()
if dst_offset + len > dst.length() {
dst.unsafe_grow_to_length(dst_offset + len)
}
Array::unsafe_blit(dst, dst_offset, self, src_offset, len)
}
///|
test "Array::blit_to/basic" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
Array::blit_to(src, dst, len=3, src_offset=1, dst_offset=2)
inspect!(dst, content="[0, 0, 2, 3, 4]")
}
///|
test "Array::blit_to/basic" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
Array::blit_to(src, dst, len=3)
inspect!(dst, content="[1, 2, 3, 0, 0]")
}
///|
test "Array::blit_to/zero_length" {
let src = [1, 2, 3]
let dst = [4, 5, 6]
Array::blit_to(src, dst, len=0)
inspect!(dst, content="[4, 5, 6]")
}
///|
test "Array::blit_to/grow_destination" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0]
Array::blit_to(src, dst, len=3, dst_offset=1)
inspect!(dst, content="[0, 1, 2, 3]")
}
///|
test "Array::blit_to/edge_cases" {
// Test with src_offset and dst_offset
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
Array::blit_to(src, dst, len=2, src_offset=1, dst_offset=2)
inspect!(dst, content="[0, 0, 2, 3, 0]")
// Test when src and dst are the same array
Array::blit_to(src, src, len=2, src_offset=0, dst_offset=3)
inspect!(src, content="[1, 2, 3, 1, 2]")
// Test with len equal to 0
Array::blit_to(src, dst, len=0, src_offset=0, dst_offset=0)
inspect!(dst, content="[0, 0, 2, 3, 0]")
// Test with len equal to the length of src
Array::blit_to(src, dst, len=5)
inspect!(dst, content="[1, 2, 3, 1, 2]")
}
///|
test "Array::blit_to/edge_cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// Test with len equal to 0
Array::blit_to(src, dst, len=0, src_offset=0, dst_offset=0)
inspect!(dst, content="[0, 0, 0, 0, 0]")
// Test with len equal to the length of src
Array::blit_to(src, dst, len=5, src_offset=0, dst_offset=0)
inspect!(dst, content="[1, 2, 3, 4, 5]")
}
///|
test "panic Array::blit_to/boundary_cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// Invalid len
ignore(Array::blit_to(src, dst, len=-1))
}
///|
test "panic Array::blit_to/boundary_cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// Invalid src_offset
ignore(Array::blit_to(src, dst, len=2, src_offset=-1))
}
///|
test "panic Array::blit_to/boundary_cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// len exceeding src length
ignore(Array::blit_to(src, dst, len=6))
}
///|
test "panic Array::blit_to/boundary_cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// dst offset exceeding dst length
ignore(Array::blit_to(src, dst, len=5, dst_offset=6))
}
///| TODO
/// 1. allow skip
/// 2. verify test
/// 3. concurrency test
test "Array::blit_to - random cases" {
let src = [10, 20, 30, 40, 50]
let dst = [0, 0, 0, 0, 0]
// Random len, src_offset, and dst_offset
Array::blit_to(src, dst, len=2, src_offset=2, dst_offset=1)
inspect!(dst, content="[0, 30, 40, 0, 0]")
// Another random case
Array::blit_to(src, dst, len=3, src_offset=1, dst_offset=2)
inspect!(dst, content="[0, 30, 20, 30, 40]")
// Yet another random case
Array::blit_to(src, dst, len=1, src_offset=4, dst_offset=4)
inspect!(dst, content="[0, 30, 20, 30, 50]")
}
///|
test "Array::blit_to - boundary cases" {
let src = [1, 2, 3, 4, 5]
let dst = [0, 0, 0, 0, 0]
// Test with src_offset at the end of src
Array::blit_to(src, dst, len=1, src_offset=4, dst_offset=0)
inspect!(dst, content="[5, 0, 0, 0, 0]")
// Test with dst_offset at the end of dst
Array::blit_to(src, dst, len=1, src_offset=0, dst_offset=4)
inspect!(dst, content="[5, 0, 0, 0, 1]")
// Test with len equal to the remaining length of src
Array::blit_to(src, dst, len=2, src_offset=3, dst_offset=3)
inspect!(dst, content="[5, 0, 0, 4, 5]")
}
///|
test "Array::unsafe_blit_fixed" {
let src = FixedArray::make(3, 1) // Create a FixedArray with 3 elements of value 1
let dst = Array::make(5, 0) // Create an Array with 5 elements of value 0
Array::unsafe_blit_fixed(dst, 1, src, 0, 2) // Copy 2 elements from src[0] to dst[1]
inspect!(dst, content="[0, 1, 1, 0, 0]")
}
// test "Array::blit_to - invalid cases" {
// let src = [1, 2, 3, 4, 5]
// let dst = [0, 0, 0, 0, 0]
// // Invalid len
// Array::blit_to(src, dst, len=-1)
// inspect!(dst, content="[0, 0, 0, 0, 0]")
// // Invalid src_offset
// Array::blit_to(src, dst, len=2, src_offset=-1)
// inspect!(dst, content="[0, 0, 0, 0, 0]")
// // Invalid dst_offset
// Array::blit_to(src, dst, len=2, dst_offset=-1)
// inspect!(dst, content="[0, 0, 0, 0, 0]")
// // len exceeding src length
// Array::blit_to(src, dst, len=6)
// inspect!(dst, content="[0, 0, 0, 0, 0]")
// // len exceeding dst length
// Array::blit_to(src, dst, len=5, dst_offset=1)
// inspect!(dst, content="[0, 0, 0, 0, 0]")
// }