This repository was archived by the owner on Aug 12, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbasalt-slicer.ads
100 lines (87 loc) · 3.29 KB
/
basalt-slicer.ads
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
--
-- @summary Generic slice generator
-- @author Johannes Kliemann
-- @date 2019-11-19
--
-- Copyright (C) 2019 Componolit GmbH
--
-- This file is part of Basalt, which is distributed under the terms of the
-- GNU Affero General Public License version 3.
--
-- Basalt.Slicer is a generic slicing tool that allows to
-- split a range into multiple subsequent slices of a maximum length.
-- All slices will be equal to the given maximum length. If the
-- range cannot be divided into full slices the last slice will be
-- smaller.
generic
type Index is range <>;
package Basalt.Slicer with
SPARK_Mode,
Pure,
Annotate => (GNATprove, Always_Return)
is
-- Slicer context
type Context is private;
-- Slice that denotes the range First .. Last
-- @field First First element of the slice
-- @field Last Last element of the slice
type Slice is record
First : Index;
Last : Index;
end record with
Dynamic_Predicate => Slice.First <= Slice.Last;
-- Create a slicer
--
-- @param Range_First First element of the range to be sliced
-- @param Range_Last Last element of the range to be sliced
-- @param Slice_Length Maximum length of a single slice
-- @return Context
function Create (Range_First : Index;
Range_Last : Index;
Slice_Length : Index) return Context with
Pre => Range_First <= Range_Last
and then Slice_Length > 0,
Post => Get_Range (Create'Result).First = Range_First
and then Get_Range (Create'Result).Last = Range_Last
and then Get_Length (Create'Result) = Slice_Length;
-- Get the current slice
--
-- @param C Context
-- @return The current slice
function Get_Slice (C : Context) return Slice with
Post => Get_Slice'Result.Last - Get_Slice'Result.First < Get_Length (C)
and then Get_Slice'Result.First in Get_Range (C).First .. Get_Range (C).Last
and then Get_Slice'Result.Last in Get_Range (C).First .. Get_Range (C).Last;
-- Get the configured range
--
-- @param C Context
-- @return The range that has been configured in Create
function Get_Range (C : Context) return Slice;
-- Get the configured length
--
-- @param C Context
-- @return The length that has been configured in Create
function Get_Length (C : Context) return Index;
-- Check if a further slice is available
--
-- @param C Context
-- @return True if a new slice is available,
-- False if the current slice is the last one
function Has_Next (C : Context) return Boolean;
-- Iterate to the next slice
--
-- @param C Context
procedure Next (C : in out Context) with
Pre => Has_Next (C),
Post => Get_Range (C'Old) = Get_Range (C)
and then Get_Length (C'Old) = Get_Length (C);
private
type Context is record
Full_Range : Slice;
Slice_Range : Slice;
Slice_Length : Index;
end record with
Dynamic_Predicate => Slice_Range.First in Full_Range.First .. Full_Range.Last
and then Slice_Range.Last in Full_Range.First .. Full_Range.Last
and then Slice_Range.Last - Slice_Range.First + 1 <= Slice_Length;
end Basalt.Slicer;