random_sample.py 6.5 KB
Newer Older
xgqdut2016's avatar
xgqdut2016 committed
1
import torch
PanZezhongQY's avatar
PanZezhongQY committed
2
import ctypes
xgqdut2016's avatar
xgqdut2016 committed
3
4
from ctypes import POINTER, Structure, c_int32, c_size_t, c_uint64, c_void_p, c_float
from libinfiniop import (
PanZezhongQY's avatar
PanZezhongQY committed
5
6
    infiniopHandle_t,
    infiniopTensorDescriptor_t,
xgqdut2016's avatar
xgqdut2016 committed
7
8
9
    open_lib,
    to_tensor,
    get_test_devices,
PanZezhongQY's avatar
PanZezhongQY committed
10
    check_error,
xgqdut2016's avatar
xgqdut2016 committed
11
    rearrange_if_needed,
PanZezhongQY's avatar
PanZezhongQY committed
12
    create_workspace,
xgqdut2016's avatar
xgqdut2016 committed
13
14
15
16
17
    test_operator,
    get_args,
    debug,
    get_tolerance,
    profile_operation,
PanZezhongQY's avatar
PanZezhongQY committed
18
19
)

xgqdut2016's avatar
xgqdut2016 committed
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
# ==============================================================================
#  Configuration (Internal Use Only)
# ==============================================================================
# These are not meant to be imported from other modules
_TEST_CASES = [
    # voc, random_val, topp, topk, temperature
        (512, 0.8, 0.8, 3, 0.5),
        (4096, 0.05, 0.9, 5, 1.0),
        (16384, 0.15, 0.85, 10, 2.0),
        (512, 0.08, 0, 3, 0.5),
        (4096, 0.5, 0.9, 1, 1.0),
        (16384, 0.15, 0, 1, 2.0),
        (16384, 0.15, 0, 1, 2.0),
        (32000, 0.08, 0.8, 50, 1.0),
        (32000, 0.08, 1.0, 25, 1.0),
        # (119696, 0.01, 1.0, 100, 1.0),
]

# Data types used for testing
_TENSOR_DTYPES = [torch.float16, torch.float32]


PROFILE = False
NUM_PRERUN = 10
NUM_ITERATIONS = 1000
PanZezhongQY's avatar
PanZezhongQY committed
45
46
47
48
49
50
51
52
53
54


class RandomSampleDescriptor(Structure):
    _fields_ = [("device", c_int32)]


infiniopRandomSampleDescriptor_t = POINTER(RandomSampleDescriptor)


def random_sample(data, random_val, topp, topk, voc, temperature, torch_device):
55
    indices = torch.zeros([topk], dtype=torch.int64)
PanZezhongQY's avatar
PanZezhongQY committed
56
57
    dataNp = data.clone().detach()
    sorted_indices = torch.arange(voc)
58

PanZezhongQY's avatar
PanZezhongQY committed
59
60
    for i in range(topk):
        for j in range(i + 1, voc):
61
            if dataNp[i] < dataNp[j]:
PanZezhongQY's avatar
PanZezhongQY committed
62
63
64
65
66
67
68
                tmp = dataNp[i].clone().detach()
                dataNp[i] = dataNp[j].clone().detach()
                dataNp[j] = tmp

                tmpInd = sorted_indices[i].clone().detach()
                sorted_indices[i] = sorted_indices[j].clone().detach()
                sorted_indices[j] = tmpInd
69
70
71
72

    # sorted_indices = torch.argsort(dataNp, descending=True)
    indices = sorted_indices[:topk]

PanZezhongQY's avatar
PanZezhongQY committed
73
    dataNp = dataNp[sorted_indices]
74

PanZezhongQY's avatar
PanZezhongQY committed
75
76
    globalM = dataNp[0]
    dataNp = (dataNp - globalM) / temperature
77
    dataNp = torch.softmax(dataNp.float(), dim=0)
PanZezhongQY's avatar
PanZezhongQY committed
78
79
80
    sum_s = 0
    for end in range(topk):
        sum_s += dataNp[end]
81
        if sum_s >= topp:
PanZezhongQY's avatar
PanZezhongQY committed
82
            break
83
    if end < topk - 1:
PanZezhongQY's avatar
PanZezhongQY committed
84
85
86
        end += 1
    else:
        end = topk
87

PanZezhongQY's avatar
PanZezhongQY committed
88
89
90
91
    sum_s = 0
    for i in range(end):
        sum_s += dataNp[i]
    random_val *= sum_s
92

PanZezhongQY's avatar
PanZezhongQY committed
93
94
95
    sum_s = 0
    for i in range(end):
        sum_s += dataNp[i]
96
        if random_val < sum_s:
PanZezhongQY's avatar
PanZezhongQY committed
97
98
            return indices[i]

99

PanZezhongQY's avatar
PanZezhongQY committed
100
101
102
def random_sample_0(data):
    return torch.argmax(data)

103
104
105
106
107
108
109
110
111
112
113
114
115

def test(
    lib,
    handle,
    torch_device,
    voc,
    random_val,
    topp,
    topk,
    temperature,
    x_dtype=torch.float16,
):
    print(f"Testing RandomSample on {torch_device} with voc:{voc} dtype:{x_dtype}")
PanZezhongQY's avatar
PanZezhongQY committed
116
117
118
    data = torch.arange(voc).float() * 0.0001
    _perm = torch.randperm(voc)
    data = data[_perm].to(x_dtype).to(torch_device)
119
120
121
122
    if topp > 0 and topk > 1:
        ans = random_sample(
            data.to("cpu"), random_val, topp, topk, voc, temperature, "cpu"
        )
PanZezhongQY's avatar
PanZezhongQY committed
123
124
125
126
127
128
129
130
131
132
    else:
        ans = random_sample_0(data)
    indices = torch.zeros([1], dtype=torch.int64).to(torch_device)
    x_tensor = to_tensor(data, lib)
    indices_tensor = to_tensor(indices, lib)
    indices_tensor.descriptor.contents.dt = U64  # treat int64 as uint64

    descriptor = infiniopRandomSampleDescriptor_t()
    check_error(
        lib.infiniopCreateRandomSampleDescriptor(
133
134
135
136
            handle,
            ctypes.byref(descriptor),
            indices_tensor.descriptor,
            x_tensor.descriptor,
PanZezhongQY's avatar
PanZezhongQY committed
137
138
139
140
        )
    )

    # Invalidate the shape and strides in the descriptor to prevent them from being directly used by the kernel
xgqdut2016's avatar
xgqdut2016 committed
141
142
    for tensor in [x_tensor, indices_tensor]:
        tensor.descriptor.contents.invalidate()
PanZezhongQY's avatar
PanZezhongQY committed
143
144
145
146
147
148
149

    workspace_size = c_uint64(0)
    check_error(
        lib.infiniopGetRandomSampleWorkspaceSize(
            descriptor, ctypes.byref(workspace_size)
        )
    )
150
    workspace = create_workspace(workspace_size.value, torch_device)
xgqdut2016's avatar
xgqdut2016 committed
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
    
    def lib_random_sample():
        check_error(
            lib.infiniopRandomSample(
                descriptor,
                workspace.data_ptr() if workspace is not None else None,
                workspace_size.value,
                indices_tensor.data,
                x_tensor.data,
                random_val,
                topp,
                topk,
                temperature,
                None,
            )
PanZezhongQY's avatar
PanZezhongQY committed
166
167
168
169
170
        )
    if torch_device == "npu":
        torch.npu.synchronize()

    assert indices[0].type(ans.dtype) == ans or data[ans] == data[indices[0]]
xgqdut2016's avatar
xgqdut2016 committed
171
172
173
174
175
176
177
178
179
180
181
182
183
    
    # Profiling workflow
    if PROFILE:
        # fmt: off
        if topp > 0 and topk > 1:
            profile_operation("PyTorch", lambda: random_sample(
                data.to("cpu"), random_val, topp, topk, voc, temperature, "cpu"
            ), torch_device, NUM_PRERUN, NUM_ITERATIONS)
        else:
            profile_operation("PyTorch", lambda: random_sample_0(data), torch_device, NUM_PRERUN, NUM_ITERATIONS)
        
        profile_operation("    lib", lambda: lib_random_sample(), torch_device, NUM_PRERUN, NUM_ITERATIONS)
        # fmt: on
PanZezhongQY's avatar
PanZezhongQY committed
184
185
    check_error(lib.infiniopDestroyRandomSampleDescriptor(descriptor))

186

PanZezhongQY's avatar
PanZezhongQY committed
187
188
189


if __name__ == "__main__":
190

PanZezhongQY's avatar
PanZezhongQY committed
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
    args = get_args()
    lib = open_lib()
    lib.infiniopCreateRandomSampleDescriptor.restype = c_int32
    lib.infiniopCreateRandomSampleDescriptor.argtypes = [
        infiniopHandle_t,
        POINTER(infiniopRandomSampleDescriptor_t),
        infiniopTensorDescriptor_t,
    ]
    lib.infiniopGetRandomSampleWorkspaceSize.restype = c_int32
    lib.infiniopGetRandomSampleWorkspaceSize.argtypes = [
        infiniopRandomSampleDescriptor_t,
        POINTER(c_uint64),
    ]
    lib.infiniopRandomSample.restype = c_int32
    lib.infiniopRandomSample.argtypes = [
        infiniopRandomSampleDescriptor_t,
        c_void_p,
        c_uint64,
        c_uint64,
        c_void_p,
        c_float,
        c_float,
        c_int32,
        c_float,
        c_void_p,
    ]
    lib.infiniopDestroyRandomSampleDescriptor.restype = c_int32
    lib.infiniopDestroyRandomSampleDescriptor.argtypes = [
        infiniopRandomSampleDescriptor_t,
    ]

xgqdut2016's avatar
xgqdut2016 committed
222
223
224
225
226
227
228
229
    PROFILE = args.profile
    NUM_PRERUN = args.num_prerun
    NUM_ITERATIONS = args.num_iterations

    # Execute tests
    for device in get_test_devices(args):
        test_operator(lib, device, test, _TEST_CASES, _TENSOR_DTYPES)

PanZezhongQY's avatar
PanZezhongQY committed
230
    print("\033[92mTest passed!\033[0m")