Check for integer overflows in places where qAllocMore is used
Task-number: QTBUG-41230 Change-Id: I5e932c2540c0bd67f13fab3ae20975d459f82c08 Reviewed-by: Thiago Macieira <thiago.macieira@intel.com> Reviewed-by: Marc Mutz <marc.mutz@kdab.com>
This commit is contained in:
parent
9eb2b25300
commit
880986be23
@ -85,8 +85,20 @@ QArrayData *QArrayData::allocate(size_t objectSize, size_t alignment,
|
|||||||
headerSize += (alignment - Q_ALIGNOF(QArrayData));
|
headerSize += (alignment - Q_ALIGNOF(QArrayData));
|
||||||
|
|
||||||
// Allocate additional space if array is growing
|
// Allocate additional space if array is growing
|
||||||
if (options & Grow)
|
if (options & Grow) {
|
||||||
capacity = qAllocMore(int(objectSize * capacity), int(headerSize)) / int(objectSize);
|
|
||||||
|
// Guard against integer overflow when multiplying.
|
||||||
|
if (capacity > std::numeric_limits<size_t>::max() / objectSize)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
size_t alloc = objectSize * capacity;
|
||||||
|
|
||||||
|
// Make sure qAllocMore won't overflow.
|
||||||
|
if (headerSize > size_t(MaxAllocSize) || alloc > size_t(MaxAllocSize) - headerSize)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
capacity = qAllocMore(int(alloc), int(headerSize)) / int(objectSize);
|
||||||
|
}
|
||||||
|
|
||||||
size_t allocSize = headerSize + objectSize * capacity;
|
size_t allocSize = headerSize + objectSize * capacity;
|
||||||
|
|
||||||
|
@ -123,7 +123,7 @@ int qFindByteArray(
|
|||||||
int qAllocMore(int alloc, int extra) Q_DECL_NOTHROW
|
int qAllocMore(int alloc, int extra) Q_DECL_NOTHROW
|
||||||
{
|
{
|
||||||
Q_ASSERT(alloc >= 0 && extra >= 0);
|
Q_ASSERT(alloc >= 0 && extra >= 0);
|
||||||
Q_ASSERT_X(alloc < (1 << 30) - extra, "qAllocMore", "Requested size is too large!");
|
Q_ASSERT_X(alloc <= MaxAllocSize - extra, "qAllocMore", "Requested size is too large!");
|
||||||
|
|
||||||
unsigned nalloc = qNextPowerOfTwo(alloc + extra);
|
unsigned nalloc = qNextPowerOfTwo(alloc + extra);
|
||||||
|
|
||||||
@ -1545,8 +1545,11 @@ void QByteArray::reallocData(uint alloc, Data::AllocationOptions options)
|
|||||||
Data::deallocate(d);
|
Data::deallocate(d);
|
||||||
d = x;
|
d = x;
|
||||||
} else {
|
} else {
|
||||||
if (options & Data::Grow)
|
if (options & Data::Grow) {
|
||||||
|
if (alloc > uint(MaxAllocSize) - uint(sizeof(Data)))
|
||||||
|
qBadAlloc();
|
||||||
alloc = qAllocMore(alloc, sizeof(Data));
|
alloc = qAllocMore(alloc, sizeof(Data));
|
||||||
|
}
|
||||||
Data *x = static_cast<Data *>(::realloc(d, sizeof(Data) + alloc));
|
Data *x = static_cast<Data *>(::realloc(d, sizeof(Data) + alloc));
|
||||||
Q_CHECK_PTR(x);
|
Q_CHECK_PTR(x);
|
||||||
x->alloc = alloc;
|
x->alloc = alloc;
|
||||||
|
@ -55,6 +55,8 @@ const QListData::Data QListData::shared_null = { Q_REFCOUNT_INITIALIZE_STATIC, 0
|
|||||||
|
|
||||||
static int grow(int size)
|
static int grow(int size)
|
||||||
{
|
{
|
||||||
|
if (size_t(size) > (MaxAllocSize - QListData::DataHeaderSize) / sizeof(void *))
|
||||||
|
qBadAlloc();
|
||||||
// dear compiler: don't optimize me out.
|
// dear compiler: don't optimize me out.
|
||||||
volatile int x = qAllocMore(size * sizeof(void *), QListData::DataHeaderSize) / sizeof(void *);
|
volatile int x = qAllocMore(size * sizeof(void *), QListData::DataHeaderSize) / sizeof(void *);
|
||||||
return x;
|
return x;
|
||||||
|
@ -1684,8 +1684,11 @@ void QString::resize(int size)
|
|||||||
|
|
||||||
void QString::reallocData(uint alloc, bool grow)
|
void QString::reallocData(uint alloc, bool grow)
|
||||||
{
|
{
|
||||||
if (grow)
|
if (grow) {
|
||||||
|
if (alloc > (uint(MaxAllocSize) - sizeof(Data)) / sizeof(QChar))
|
||||||
|
qBadAlloc();
|
||||||
alloc = qAllocMore(alloc * sizeof(QChar), sizeof(Data)) / sizeof(QChar);
|
alloc = qAllocMore(alloc * sizeof(QChar), sizeof(Data)) / sizeof(QChar);
|
||||||
|
}
|
||||||
|
|
||||||
if (d->ref.isShared() || IS_RAW_DATA(d)) {
|
if (d->ref.isShared() || IS_RAW_DATA(d)) {
|
||||||
Data::AllocationOptions allocOptions(d->capacityReserved ? Data::CapacityReserved : 0);
|
Data::AllocationOptions allocOptions(d->capacityReserved ? Data::CapacityReserved : 0);
|
||||||
|
@ -46,9 +46,15 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
#include "QtCore/qglobal.h"
|
#include "QtCore/qglobal.h"
|
||||||
|
#include <limits>
|
||||||
|
|
||||||
QT_BEGIN_NAMESPACE
|
QT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
|
// We typically need an extra bit for qNextPowerOfTwo when determining the next allocation size.
|
||||||
|
enum {
|
||||||
|
MaxAllocSize = (1 << (std::numeric_limits<int>::digits - 1)) - 1
|
||||||
|
};
|
||||||
|
|
||||||
// implemented in qbytearray.cpp
|
// implemented in qbytearray.cpp
|
||||||
int Q_CORE_EXPORT qAllocMore(int alloc, int extra) Q_DECL_NOTHROW;
|
int Q_CORE_EXPORT qAllocMore(int alloc, int extra) Q_DECL_NOTHROW;
|
||||||
|
|
||||||
|
@ -249,6 +249,8 @@ uint QFragmentMapData<Fragment>::createFragment()
|
|||||||
uint freePos = head->freelist;
|
uint freePos = head->freelist;
|
||||||
if (freePos == head->allocated) {
|
if (freePos == head->allocated) {
|
||||||
// need to create some free space
|
// need to create some free space
|
||||||
|
if (freePos >= uint(MaxAllocSize) / fragmentSize)
|
||||||
|
qBadAlloc();
|
||||||
uint needed = qAllocMore((freePos+1)*fragmentSize, 0);
|
uint needed = qAllocMore((freePos+1)*fragmentSize, 0);
|
||||||
Q_ASSERT(needed/fragmentSize > head->allocated);
|
Q_ASSERT(needed/fragmentSize > head->allocated);
|
||||||
Fragment *newFragments = (Fragment *)realloc(fragments, needed);
|
Fragment *newFragments = (Fragment *)realloc(fragments, needed);
|
||||||
|
Loading…
Reference in New Issue
Block a user