Skip to content

Commit 815d383

Browse files
peterjc123facebook-github-bot
authored andcommitted
PyLong_{As/From}{Long/UnsignedLong} lint checks (pytorch#49280)
Summary: Fixes pytorch#45581 Pull Request resolved: pytorch#49280 Reviewed By: mruberry Differential Revision: D25592330 Pulled By: ezyang fbshipit-source-id: 5c16d6aed88ad1feaa7f129b4cd44c0561be2de2
1 parent c20b916 commit 815d383

File tree

9 files changed

+84
-26
lines changed

9 files changed

+84
-26
lines changed

.jenkins/pytorch/win-build.sh

+15
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,21 @@ fi
3838

3939
export SCRIPT_HELPERS_DIR=$SCRIPT_PARENT_DIR/win-test-helpers
4040

41+
set +ex
42+
grep -E -R 'PyLong_(From|As)(Unsigned|)Long\(' --exclude=python_numbers.h torch/
43+
PYLONG_API_CHECK=$?
44+
if [[ $PYLONG_API_CHECK == 0 ]]; then
45+
echo "Usage of PyLong_{From,As}{Unsigned}Long API may lead to overflow errors on Windows"
46+
echo "because \`sizeof(long) == 4\` and \`sizeof(unsigned long) == 4\`."
47+
echo "Please include \"torch/csrc/python_numbers.h\" and use the correspoding APIs instead."
48+
echo "PyLong_FromLong -> THPUtils_packInt32 / THPUtils_packInt64"
49+
echo "PyLong_AsLong -> THPUtils_unpackInt (32-bit) / THPUtils_unpackLong (64-bit)"
50+
echo "PyLong_FromUnsignedLong -> THPUtils_packUInt32 / THPUtils_packUInt64"
51+
echo "PyLong_AsUnsignedLong -> THPUtils_unpackUInt32 / THPUtils_unpackUInt64"
52+
exit 1
53+
fi
54+
set -ex
55+
4156
$SCRIPT_HELPERS_DIR/build_pytorch.bat
4257

4358
assert_git_not_dirty

torch/csrc/Module.cpp

+8-7
Original file line numberDiff line numberDiff line change
@@ -153,27 +153,28 @@ static PyObject * THPModule_crashIfCsrcASAN(PyObject *module, PyObject *arg) {
153153
"but got %s", THPUtils_typename(arg));
154154
//NOLINTNEXTLINE(cppcoreguidelines-avoid-c-arrays, modernize-avoid-c-arrays)
155155
volatile char x[3];
156-
x[static_cast<int>(THPUtils_unpackLong(arg))] = 0;
157-
return PyLong_FromLong(x[0]);
156+
x[THPUtils_unpackInt(arg)] = 0;
157+
//NOLINTNEXTLINE(clang-analyzer-core.CallAndMessage)
158+
return THPUtils_packInt32(x[0]);
158159
}
159160

160161
static PyObject * THPModule_crashIfCsrcUBSAN(PyObject *module, PyObject *arg) {
161162
THPUtils_assert(THPUtils_checkLong(arg), "crash_if_csrc_ubsan expects an int, "
162163
"but got %s", THPUtils_typename(arg));
163-
int32_t x = static_cast<int>(THPUtils_unpackLong(arg));
164+
int32_t x = THPUtils_unpackInt(arg);
164165
double y = 1.0 / x;
165-
return PyLong_FromLong((int)y);
166+
return THPUtils_packInt32((int)y);
166167
}
167168

168169
static PyObject * THPModule_crashIfATenASAN(PyObject *module, PyObject *arg) {
169170
THPUtils_assert(THPUtils_checkLong(arg), "crash_if_aten_asan expects an int, "
170171
"but got %s", THPUtils_typename(arg));
171-
return PyLong_FromLong(at::_crash_if_asan(static_cast<int>(THPUtils_unpackLong(arg))));
172+
return THPUtils_packInt32(at::_crash_if_asan(THPUtils_unpackInt(arg)));
172173
}
173174

174175
static PyObject * THPModule_getNumThreads(PyObject *module, PyObject *noargs)
175176
{
176-
return PyLong_FromLong(at::get_num_threads());
177+
return THPUtils_packInt32(at::get_num_threads());
177178
}
178179

179180
static PyObject * THPModule_setNumThreads(PyObject *module, PyObject *arg)
@@ -188,7 +189,7 @@ static PyObject * THPModule_setNumThreads(PyObject *module, PyObject *arg)
188189

189190
static PyObject * THPModule_getNumInteropThreads(PyObject *module, PyObject *noargs)
190191
{
191-
return PyLong_FromLong(at::get_num_interop_threads());
192+
return THPUtils_packInt32(at::get_num_interop_threads());
192193
}
193194

194195
static PyObject * THPModule_setNumInteropThreads(PyObject *module, PyObject *arg)

torch/csrc/Size.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
#include <string>
44
#include <torch/csrc/utils/object_ptr.h>
5+
#include <torch/csrc/utils/python_numbers.h>
56
#include <torch/csrc/utils/python_strings.h>
67
#include <torch/csrc/utils/python_tuples.h>
78

@@ -84,7 +85,7 @@ static PyObject * THPSize_repr(THPSize *self)
8485
if (i != 0) {
8586
repr += ", ";
8687
}
87-
repr += std::to_string(PyLong_AsLong(PyTuple_GET_ITEM(self, i)));
88+
repr += std::to_string(THPUtils_unpackLong(PyTuple_GET_ITEM(self, i)));
8889
}
8990
repr += "])";
9091
return THPUtils_packString(repr);
@@ -136,7 +137,7 @@ static PyObject *THPSize_numel(PyObject *_self, PyObject *noargs)
136137
auto self = (THPSize*)_self;
137138
int64_t numel = 1;
138139
for (Py_ssize_t i = 0; i < PyTuple_Size((PyObject*)self); ++i) {
139-
numel *= PyLong_AsLong(PyTuple_GET_ITEM(self, i));
140+
numel *= THPUtils_unpackLong(PyTuple_GET_ITEM(self, i));
140141
}
141142
return THPUtils_packInt64(numel);
142143
END_HANDLE_TH_ERRORS

torch/csrc/autograd/python_cpp_function.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
#include <torch/csrc/autograd/python_hook.h>
1212
#include <torch/csrc/autograd/python_anomaly_mode.h>
1313
#include <pybind11/pybind11.h>
14+
#include <torch/csrc/utils/python_numbers.h>
1415
#include <torch/csrc/utils/python_strings.h>
1516
#include <torch/csrc/DynamicTypes.h>
1617
#include <torch/csrc/Exceptions.h>
@@ -114,7 +115,7 @@ PyObject* THPCppFunction_next_functions(THPCppFunction* self, PyObject* hook)
114115
PyObject *py_fn = functionToPyObject(c_tuple.function);
115116
if (!py_fn) return nullptr;
116117
PyTuple_SET_ITEM(tuple.get(), 0, py_fn);
117-
PyObject *py_idx = PyLong_FromLong(c_tuple.input_nr);
118+
PyObject *py_idx = THPUtils_packUInt32(c_tuple.input_nr);
118119
if (!py_idx) return nullptr;
119120
PyTuple_SET_ITEM(tuple.get(), 1, py_idx);
120121
PyTuple_SET_ITEM(py_functions.get(), i, tuple.release());

torch/csrc/cuda/Module.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
#include <torch/csrc/CudaIPCTypes.h>
1818
#include <torch/csrc/utils/pybind.h>
1919
#include <torch/csrc/utils/cuda_lazy_init.h>
20+
#include <torch/csrc/utils/python_numbers.h>
2021
#include <torch/csrc/utils/python_strings.h>
2122
#include <torch/csrc/cuda/python_comm.h>
2223
#include <torch/csrc/Generator.h>
@@ -77,15 +78,15 @@ PyObject * THCPModule_getDevice_wrap(PyObject *self, PyObject *noargs)
7778
HANDLE_TH_ERRORS
7879
torch::utils::cuda_lazy_init();
7980
auto device = static_cast<int>(c10::cuda::current_device());
80-
return PyLong_FromLong(device);
81+
return THPUtils_packInt32(device);
8182
END_HANDLE_TH_ERRORS
8283
}
8384

8485
PyObject * THCPModule_getDeviceCount_wrap(PyObject *self, PyObject *noargs)
8586
{
8687
HANDLE_TH_ERRORS
8788
poison_fork();
88-
return PyLong_FromLong(at::cuda::device_count());
89+
return THPUtils_packUInt64(at::cuda::device_count());
8990
END_HANDLE_TH_ERRORS
9091
}
9192

@@ -150,7 +151,7 @@ PyObject * THCPModule_setStream_wrap(PyObject *self, PyObject *obj)
150151

151152
PyObject * THCPModule_getCompiledVersion(PyObject *self, PyObject *noargs)
152153
{
153-
return PyLong_FromLong((long) CUDA_VERSION);
154+
return THPUtils_packInt64((int64_t) CUDA_VERSION);
154155
}
155156

156157
PyObject * THCPModule_cudaHostAllocator(PyObject *_unused, PyObject *noargs)

torch/csrc/cuda/Stream.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#include <pybind11/pybind11.h>
22
#include <torch/csrc/cuda/Stream.h>
33
#include <torch/csrc/cuda/Module.h>
4+
#include <torch/csrc/utils/python_numbers.h>
45
#include <torch/csrc/Device.h>
56
#include <torch/csrc/THP.h>
67

@@ -65,7 +66,7 @@ static PyObject * THCPStream_get_cuda_stream(THCPStream *self, void *unused) {
6566

6667
static PyObject * THCPStream_get_priority(THCPStream *self, void *unused) {
6768
HANDLE_TH_ERRORS
68-
return PyLong_FromLong(self->cuda_stream.priority());
69+
return THPUtils_packInt64(self->cuda_stream.priority());
6970
END_HANDLE_TH_ERRORS
7071
}
7172

torch/csrc/generic/StorageMethods.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
#include <ATen/ATen.h>
22
#include <torch/csrc/utils/pycfunction_helpers.h>
3+
#include <torch/csrc/utils/python_numbers.h>
34

45
#ifdef USE_CUDA
56
#include <cuda_runtime.h>
@@ -15,7 +16,7 @@ static PyObject * THPStorage_(size)(PyObject *_self, PyObject *noargs)
1516
{
1617
HANDLE_TH_ERRORS
1718
auto self = (THPStorage*)_self;
18-
return PyLong_FromLong(self->cdata->nbytes() / sizeof(scalar_t));
19+
return THPUtils_packUInt64(self->cdata->nbytes() / sizeof(scalar_t));
1920
END_HANDLE_TH_ERRORS
2021
}
2122

@@ -50,7 +51,7 @@ static PyObject * THPStorage_(elementSize)(PyObject *_self, PyObject *noargs)
5051
{
5152
HANDLE_TH_ERRORS
5253
auto self = (THPStorage*)_self;
53-
return PyLong_FromLong(THWStorage_(elementSize)(LIBRARY_STATE_NOARGS));
54+
return THPUtils_packInt64(THWStorage_(elementSize)(LIBRARY_STATE_NOARGS));
5455
END_HANDLE_TH_ERRORS
5556
}
5657

@@ -315,7 +316,7 @@ PyObject * THPStorage_(getDevice)(PyObject *_self, PyObject *noargs)
315316
{
316317
HANDLE_TH_ERRORS
317318
auto self = (THPStorage*)_self;
318-
return PyLong_FromLong(THCStorage_(getDevice)(LIBRARY_STATE self->cdata));
319+
return THPUtils_packInt32(THCStorage_(getDevice)(LIBRARY_STATE self->cdata));
319320
END_HANDLE_TH_ERRORS
320321
}
321322
#endif

torch/csrc/generic/StorageSharing.cpp

+10-9
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
#include <c10/cuda/CUDAGuard.h>
55
#endif
66

7+
#include <torch/csrc/utils/python_numbers.h>
78
#include <random>
89

910
static PyObject * THPStorage_(sharedDecref)(PyObject *_self, PyObject *noargs)
@@ -95,7 +96,7 @@ static PyObject * THPStorage_(shareFilename)(PyObject *_self, PyObject *noargs)
9596
if (!manager_handle) return nullptr;
9697
THPObjectPtr storage_handle(PyBytes_FromString(ctx->filename()));
9798
if (!storage_handle) return nullptr;
98-
THPObjectPtr size(PyLong_FromLong(storage->nbytes() / sizeof(scalar_t)));
99+
THPObjectPtr size(THPUtils_packUInt64(storage->nbytes() / sizeof(scalar_t)));
99100
if (!size) return nullptr;
100101

101102
THPObjectPtr tuple(PyTuple_New(3));
@@ -172,9 +173,9 @@ static PyObject * THPStorage_(shareFd)(PyObject *_self, PyObject *noargs)
172173
AT_ASSERT(ctx);
173174
}
174175

175-
THPObjectPtr storage_handle(PyLong_FromLong(ctx->fd()));
176+
THPObjectPtr storage_handle(THPUtils_packInt32(ctx->fd()));
176177
if (!storage_handle) return nullptr;
177-
THPObjectPtr size(PyLong_FromLong(storage->nbytes() / sizeof(scalar_t)));
178+
THPObjectPtr size(THPUtils_packUInt64(storage->nbytes() / sizeof(scalar_t)));
178179
if (!size) return nullptr;
179180

180181
THPObjectPtr tuple(PyTuple_New(2));
@@ -231,14 +232,14 @@ static PyObject * THPStorage_(shareCuda)(PyObject *_self, PyObject *noargs)
231232

232233
at::DeviceGuard device_guard(storage->device());
233234
THPObjectPtr tuple(PyTuple_New(8));
234-
THPObjectPtr device(PyLong_FromLong(storage->device().index()));
235+
THPObjectPtr device(THPUtils_packInt32(storage->device().index()));
235236
THPObjectPtr _handle(Py_None);
236237
Py_INCREF(Py_None);
237-
THPObjectPtr size_bytes(PyLong_FromLong(storage->nbytes()));
238-
THPObjectPtr _offset_bytes(PyLong_FromLong(0));
238+
THPObjectPtr size_bytes(THPUtils_packUInt64(storage->nbytes()));
239+
THPObjectPtr _offset_bytes(THPUtils_packInt32(0));
239240
THPObjectPtr _ref_counter(Py_None);
240241
Py_INCREF(Py_None);
241-
THPObjectPtr _ref_counter_offset(PyLong_FromLong(0));
242+
THPObjectPtr _ref_counter_offset(THPUtils_packInt32(0));
242243
THPObjectPtr _event_handle(Py_None);
243244
Py_INCREF(Py_None);
244245
THPObjectPtr _event_sync_required(Py_None);
@@ -261,7 +262,7 @@ static PyObject * THPStorage_(shareCuda)(PyObject *_self, PyObject *noargs)
261262
auto sent_data = static_cast<torch::CudaIPCSentData*>(storage->data_ptr().get_context());
262263
sent_data->set_original_ptr(std::move(old_data_ptr));
263264
_ref_counter = PyBytes_FromString((sent_data->handle()).c_str());
264-
_ref_counter_offset = PyLong_FromLong(sent_data->offset());
265+
_ref_counter_offset = THPUtils_packInt64(sent_data->offset());
265266

266267

267268
cudaIpcEventHandle_t ipc_event_handle;
@@ -515,7 +516,7 @@ PyObject * THPStorage_(sharedFd)(PyObject *_self, PyObject *noargs)
515516
#endif
516517

517518
THPUtils_assert(ctx, "couldn't retrieve a shared file descriptor");
518-
return PyLong_FromLong(ctx->fd());
519+
return THPUtils_packInt32(ctx->fd());
519520
END_HANDLE_TH_ERRORS
520521
}
521522

torch/csrc/utils/python_numbers.h

+36
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,24 @@
66
#include <torch/csrc/utils/object_ptr.h>
77
#include <torch/csrc/utils/tensor_numpy.h>
88
#include <cstdint>
9+
#include <limits>
910
#include <stdexcept>
1011

1112
// largest integer that can be represented consecutively in a double
1213
const int64_t DOUBLE_INT_MAX = 9007199254740992;
1314

15+
inline PyObject* THPUtils_packInt32(int32_t value) {
16+
return PyLong_FromLong(value);
17+
}
18+
1419
inline PyObject* THPUtils_packInt64(int64_t value) {
1520
return PyLong_FromLongLong(value);
1621
}
1722

23+
inline PyObject* THPUtils_packUInt32(uint32_t value) {
24+
return PyLong_FromUnsignedLong(value);
25+
}
26+
1827
inline PyObject* THPUtils_packUInt64(uint64_t value) {
1928
return PyLong_FromUnsignedLongLong(value);
2029
}
@@ -33,6 +42,22 @@ inline bool THPUtils_checkLong(PyObject* obj) {
3342
return PyLong_Check(obj) && !PyBool_Check(obj);
3443
}
3544

45+
inline int32_t THPUtils_unpackInt(PyObject* obj) {
46+
int overflow;
47+
long value = PyLong_AsLongAndOverflow(obj, &overflow);
48+
if (value == -1 && PyErr_Occurred()) {
49+
throw python_error();
50+
}
51+
if (overflow != 0) {
52+
throw std::runtime_error("Overflow when unpacking long");
53+
}
54+
if (value > std::numeric_limits<int32_t>::max() ||
55+
value < std::numeric_limits<int32_t>::min()) {
56+
throw std::runtime_error("Overflow when unpacking long");
57+
}
58+
return (int32_t)value;
59+
}
60+
3661
inline int64_t THPUtils_unpackLong(PyObject* obj) {
3762
int overflow;
3863
long long value = PyLong_AsLongLongAndOverflow(obj, &overflow);
@@ -45,6 +70,17 @@ inline int64_t THPUtils_unpackLong(PyObject* obj) {
4570
return (int64_t)value;
4671
}
4772

73+
inline uint32_t THPUtils_unpackUInt32(PyObject* obj) {
74+
unsigned long value = PyLong_AsUnsignedLong(obj);
75+
if (PyErr_Occurred()) {
76+
throw python_error();
77+
}
78+
if (value > std::numeric_limits<uint32_t>::max()) {
79+
throw std::runtime_error("Overflow when unpacking unsigned long");
80+
}
81+
return (uint32_t)value;
82+
}
83+
4884
inline uint64_t THPUtils_unpackUInt64(PyObject* obj) {
4985
unsigned long long value = PyLong_AsUnsignedLongLong(obj);
5086
if (PyErr_Occurred()) {

0 commit comments

Comments
 (0)