2017-12-21 07:45:38 +08:00
|
|
|
/*
|
2018-06-13 03:54:39 +08:00
|
|
|
* Copyright (c) 2017 - 2018, Intel Corporation
|
2017-12-21 07:45:38 +08:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included
|
|
|
|
* in all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
|
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
|
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#include "runtime/os_interface/32bit_memory.h"
|
|
|
|
#include "runtime/os_interface/linux/drm_32bit_memory.cpp"
|
|
|
|
#include "runtime/helpers/aligned_memory.h"
|
|
|
|
namespace OCLRT {
|
|
|
|
|
|
|
|
constexpr uintptr_t startOf32MmapRegion = 0x40000000;
|
|
|
|
static bool failMmap = false;
|
|
|
|
static bool fail32BitMmap = false;
|
|
|
|
static bool failUpperRange = false;
|
|
|
|
static bool failLowerRanger = false;
|
|
|
|
|
|
|
|
static uintptr_t startUpperHeap = maxMmap32BitAddress;
|
|
|
|
static uintptr_t lowerRangeHeapStart = lowerRangeStart;
|
|
|
|
|
|
|
|
static uintptr_t offsetIn32BitRange = 0;
|
|
|
|
static uint32_t mmapCallCount = 0u;
|
|
|
|
static uint32_t unmapCallCount = 0u;
|
|
|
|
static uint32_t mmapFailCount = 0u;
|
|
|
|
|
|
|
|
void *MockMmap(void *addr, size_t length, int prot, int flags,
|
|
|
|
int fd, off_t offset) noexcept {
|
|
|
|
|
|
|
|
bool return32bitRange = true;
|
|
|
|
bool returnUpperRange = false;
|
|
|
|
bool returnLowerRange = false;
|
|
|
|
mmapCallCount++;
|
|
|
|
|
|
|
|
if (failMmap)
|
|
|
|
return MAP_FAILED;
|
|
|
|
|
|
|
|
if (mmapFailCount > 0) {
|
|
|
|
mmapFailCount--;
|
|
|
|
return MAP_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addr) {
|
|
|
|
return32bitRange = false;
|
|
|
|
if ((uintptr_t)addr >= maxMmap32BitAddress) {
|
|
|
|
if (failUpperRange) {
|
|
|
|
return MAP_FAILED;
|
|
|
|
}
|
|
|
|
returnUpperRange = true;
|
|
|
|
}
|
|
|
|
if ((uintptr_t)addr >= lowerRangeStart) {
|
|
|
|
if (failLowerRanger) {
|
|
|
|
return MAP_FAILED;
|
|
|
|
}
|
|
|
|
returnLowerRange = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & MAP_32BIT) {
|
|
|
|
if (fail32BitMmap) {
|
|
|
|
return MAP_FAILED;
|
|
|
|
}
|
|
|
|
return32bitRange = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uintptr_t ptrToReturn = (uintptr_t)addr;
|
|
|
|
if (return32bitRange) {
|
|
|
|
ptrToReturn = startOf32MmapRegion + offsetIn32BitRange;
|
|
|
|
offsetIn32BitRange += alignUp(length, MemoryConstants::pageSize);
|
|
|
|
} else if (returnUpperRange) {
|
|
|
|
ptrToReturn = (uintptr_t)addr;
|
|
|
|
} else if (returnLowerRange) {
|
|
|
|
ptrToReturn = (uintptr_t)addr;
|
|
|
|
} else {
|
|
|
|
ptrToReturn = (uintptr_t)MAP_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (void *)ptrToReturn;
|
|
|
|
}
|
|
|
|
int MockMunmap(void *addr, size_t length) noexcept {
|
|
|
|
unmapCallCount++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
class mockAllocator32Bit : public Allocator32bit {
|
|
|
|
public:
|
|
|
|
class OsInternalsPublic : public Allocator32bit::OsInternals {
|
|
|
|
};
|
|
|
|
|
|
|
|
mockAllocator32Bit(Allocator32bit::OsInternals *osInternalsIn) : Allocator32bit(osInternalsIn) {
|
|
|
|
}
|
|
|
|
|
|
|
|
mockAllocator32Bit() {
|
|
|
|
this->osInternals->mmapFunction = MockMmap;
|
|
|
|
this->osInternals->munmapFunction = MockMunmap;
|
|
|
|
resetState();
|
|
|
|
}
|
|
|
|
~mockAllocator32Bit() {
|
|
|
|
resetState();
|
|
|
|
}
|
|
|
|
static void resetState() {
|
|
|
|
fail32BitMmap = false;
|
|
|
|
failUpperRange = false;
|
|
|
|
failLowerRanger = false;
|
|
|
|
failMmap = false;
|
|
|
|
startUpperHeap = maxMmap32BitAddress;
|
|
|
|
lowerRangeHeapStart = lowerRangeStart;
|
|
|
|
offsetIn32BitRange = 0u;
|
|
|
|
mmapCallCount = 0u;
|
|
|
|
unmapCallCount = 0u;
|
|
|
|
mmapFailCount = 0u;
|
|
|
|
}
|
|
|
|
|
|
|
|
static OsInternalsPublic *createOsInternals() {
|
|
|
|
return new OsInternalsPublic;
|
|
|
|
}
|
|
|
|
|
|
|
|
OsInternals *getosInternal() { return this->osInternals.get(); }
|
|
|
|
};
|
2018-06-13 03:54:39 +08:00
|
|
|
} // namespace OCLRT
|