提交 e3ae90a5 编写于 作者: A acorn

Merge

...@@ -2269,15 +2269,16 @@ void linux_wrap_code(char* base, size_t size) { ...@@ -2269,15 +2269,16 @@ void linux_wrap_code(char* base, size_t size) {
// All it does is to check if there are enough free pages // All it does is to check if there are enough free pages
// left at the time of mmap(). This could be a potential // left at the time of mmap(). This could be a potential
// problem. // problem.
bool os::commit_memory(char* addr, size_t size) { bool os::commit_memory(char* addr, size_t size, bool exec) {
uintptr_t res = (uintptr_t) ::mmap(addr, size, int prot = exec ? PROT_READ|PROT_WRITE|PROT_EXEC : PROT_READ|PROT_WRITE;
PROT_READ|PROT_WRITE|PROT_EXEC, uintptr_t res = (uintptr_t) ::mmap(addr, size, prot,
MAP_PRIVATE|MAP_FIXED|MAP_ANONYMOUS, -1, 0); MAP_PRIVATE|MAP_FIXED|MAP_ANONYMOUS, -1, 0);
return res != (uintptr_t) MAP_FAILED; return res != (uintptr_t) MAP_FAILED;
} }
bool os::commit_memory(char* addr, size_t size, size_t alignment_hint) { bool os::commit_memory(char* addr, size_t size, size_t alignment_hint,
return commit_memory(addr, size); bool exec) {
return commit_memory(addr, size, exec);
} }
void os::realign_memory(char *addr, size_t bytes, size_t alignment_hint) { } void os::realign_memory(char *addr, size_t bytes, size_t alignment_hint) { }
...@@ -2417,8 +2418,7 @@ os::Linux::numa_interleave_memory_func_t os::Linux::_numa_interleave_memory; ...@@ -2417,8 +2418,7 @@ os::Linux::numa_interleave_memory_func_t os::Linux::_numa_interleave_memory;
unsigned long* os::Linux::_numa_all_nodes; unsigned long* os::Linux::_numa_all_nodes;
bool os::uncommit_memory(char* addr, size_t size) { bool os::uncommit_memory(char* addr, size_t size) {
return ::mmap(addr, size, return ::mmap(addr, size, PROT_NONE,
PROT_READ|PROT_WRITE|PROT_EXEC,
MAP_PRIVATE|MAP_FIXED|MAP_NORESERVE|MAP_ANONYMOUS, -1, 0) MAP_PRIVATE|MAP_FIXED|MAP_NORESERVE|MAP_ANONYMOUS, -1, 0)
!= MAP_FAILED; != MAP_FAILED;
} }
...@@ -2441,7 +2441,9 @@ static char* anon_mmap(char* requested_addr, size_t bytes, bool fixed) { ...@@ -2441,7 +2441,9 @@ static char* anon_mmap(char* requested_addr, size_t bytes, bool fixed) {
flags |= MAP_FIXED; flags |= MAP_FIXED;
} }
addr = (char*)::mmap(requested_addr, bytes, PROT_READ|PROT_WRITE|PROT_EXEC, // Map uncommitted pages PROT_READ and PROT_WRITE, change access
// to PROT_EXEC if executable when we commit the page.
addr = (char*)::mmap(requested_addr, bytes, PROT_READ|PROT_WRITE,
flags, -1, 0); flags, -1, 0);
if (addr != MAP_FAILED) { if (addr != MAP_FAILED) {
...@@ -2582,7 +2584,9 @@ bool os::large_page_init() { ...@@ -2582,7 +2584,9 @@ bool os::large_page_init() {
#define SHM_HUGETLB 04000 #define SHM_HUGETLB 04000
#endif #endif
char* os::reserve_memory_special(size_t bytes, char* req_addr) { char* os::reserve_memory_special(size_t bytes, char* req_addr, bool exec) {
// "exec" is passed in but not used. Creating the shared image for
// the code cache doesn't have an SHM_X executable permission to check.
assert(UseLargePages, "only for large pages"); assert(UseLargePages, "only for large pages");
key_t key = IPC_PRIVATE; key_t key = IPC_PRIVATE;
......
...@@ -2623,15 +2623,16 @@ int os::vm_allocation_granularity() { ...@@ -2623,15 +2623,16 @@ int os::vm_allocation_granularity() {
return page_size; return page_size;
} }
bool os::commit_memory(char* addr, size_t bytes) { bool os::commit_memory(char* addr, size_t bytes, bool exec) {
int prot = exec ? PROT_READ|PROT_WRITE|PROT_EXEC : PROT_READ|PROT_WRITE;
size_t size = bytes; size_t size = bytes;
return return
NULL != Solaris::mmap_chunk(addr, size, MAP_PRIVATE|MAP_FIXED, NULL != Solaris::mmap_chunk(addr, size, MAP_PRIVATE|MAP_FIXED, prot);
PROT_READ | PROT_WRITE | PROT_EXEC);
} }
bool os::commit_memory(char* addr, size_t bytes, size_t alignment_hint) { bool os::commit_memory(char* addr, size_t bytes, size_t alignment_hint,
if (commit_memory(addr, bytes)) { bool exec) {
if (commit_memory(addr, bytes, exec)) {
if (UseMPSS && alignment_hint > (size_t)vm_page_size()) { if (UseMPSS && alignment_hint > (size_t)vm_page_size()) {
// If the large page size has been set and the VM // If the large page size has been set and the VM
// is using large pages, use the large page size // is using large pages, use the large page size
...@@ -3220,7 +3221,9 @@ bool os::Solaris::set_mpss_range(caddr_t start, size_t bytes, size_t align) { ...@@ -3220,7 +3221,9 @@ bool os::Solaris::set_mpss_range(caddr_t start, size_t bytes, size_t align) {
return true; return true;
} }
char* os::reserve_memory_special(size_t bytes, char* addr) { char* os::reserve_memory_special(size_t bytes, char* addr, bool exec) {
// "exec" is passed in but not used. Creating the shared image for
// the code cache doesn't have an SHM_X executable permission to check.
assert(UseLargePages && UseISM, "only for ISM large pages"); assert(UseLargePages && UseISM, "only for ISM large pages");
size_t size = bytes; size_t size = bytes;
......
...@@ -2189,7 +2189,8 @@ LONG WINAPI topLevelExceptionFilter(struct _EXCEPTION_POINTERS* exceptionInfo) { ...@@ -2189,7 +2189,8 @@ LONG WINAPI topLevelExceptionFilter(struct _EXCEPTION_POINTERS* exceptionInfo) {
if (addr > thread->stack_yellow_zone_base() && addr < thread->stack_base() ) { if (addr > thread->stack_yellow_zone_base() && addr < thread->stack_base() ) {
addr = (address)((uintptr_t)addr & addr = (address)((uintptr_t)addr &
(~((uintptr_t)os::vm_page_size() - (uintptr_t)1))); (~((uintptr_t)os::vm_page_size() - (uintptr_t)1)));
os::commit_memory( (char *)addr, thread->stack_base() - addr ); os::commit_memory((char *)addr, thread->stack_base() - addr,
false );
return EXCEPTION_CONTINUE_EXECUTION; return EXCEPTION_CONTINUE_EXECUTION;
} }
else else
...@@ -2565,8 +2566,7 @@ char* os::reserve_memory(size_t bytes, char* addr, size_t alignment_hint) { ...@@ -2565,8 +2566,7 @@ char* os::reserve_memory(size_t bytes, char* addr, size_t alignment_hint) {
assert((size_t)addr % os::vm_allocation_granularity() == 0, assert((size_t)addr % os::vm_allocation_granularity() == 0,
"reserve alignment"); "reserve alignment");
assert(bytes % os::vm_allocation_granularity() == 0, "reserve block size"); assert(bytes % os::vm_allocation_granularity() == 0, "reserve block size");
char* res = (char*)VirtualAlloc(addr, bytes, MEM_RESERVE, char* res = (char*)VirtualAlloc(addr, bytes, MEM_RESERVE, PAGE_READWRITE);
PAGE_EXECUTE_READWRITE);
assert(res == NULL || addr == NULL || addr == res, assert(res == NULL || addr == NULL || addr == res,
"Unexpected address from reserve."); "Unexpected address from reserve.");
return res; return res;
...@@ -2595,7 +2595,7 @@ bool os::can_execute_large_page_memory() { ...@@ -2595,7 +2595,7 @@ bool os::can_execute_large_page_memory() {
return true; return true;
} }
char* os::reserve_memory_special(size_t bytes, char* addr) { char* os::reserve_memory_special(size_t bytes, char* addr, bool exec) {
if (UseLargePagesIndividualAllocation) { if (UseLargePagesIndividualAllocation) {
if (TracePageSizes && Verbose) { if (TracePageSizes && Verbose) {
...@@ -2618,7 +2618,7 @@ char* os::reserve_memory_special(size_t bytes, char* addr) { ...@@ -2618,7 +2618,7 @@ char* os::reserve_memory_special(size_t bytes, char* addr) {
p_buf = (char *) VirtualAlloc(addr, p_buf = (char *) VirtualAlloc(addr,
size_of_reserve, // size of Reserve size_of_reserve, // size of Reserve
MEM_RESERVE, MEM_RESERVE,
PAGE_EXECUTE_READWRITE); PAGE_READWRITE);
// If reservation failed, return NULL // If reservation failed, return NULL
if (p_buf == NULL) return NULL; if (p_buf == NULL) return NULL;
...@@ -2659,7 +2659,13 @@ char* os::reserve_memory_special(size_t bytes, char* addr) { ...@@ -2659,7 +2659,13 @@ char* os::reserve_memory_special(size_t bytes, char* addr) {
p_new = (char *) VirtualAlloc(next_alloc_addr, p_new = (char *) VirtualAlloc(next_alloc_addr,
bytes_to_rq, bytes_to_rq,
MEM_RESERVE | MEM_COMMIT | MEM_LARGE_PAGES, MEM_RESERVE | MEM_COMMIT | MEM_LARGE_PAGES,
PAGE_EXECUTE_READWRITE); PAGE_READWRITE);
if (p_new != NULL && exec) {
DWORD oldprot;
// Windows doc says to use VirtualProtect to get execute permissions
VirtualProtect(next_alloc_addr, bytes_to_rq,
PAGE_EXECUTE_READWRITE, &oldprot);
}
} }
if (p_new == NULL) { if (p_new == NULL) {
...@@ -2688,10 +2694,12 @@ char* os::reserve_memory_special(size_t bytes, char* addr) { ...@@ -2688,10 +2694,12 @@ char* os::reserve_memory_special(size_t bytes, char* addr) {
} else { } else {
// normal policy just allocate it all at once // normal policy just allocate it all at once
DWORD flag = MEM_RESERVE | MEM_COMMIT | MEM_LARGE_PAGES; DWORD flag = MEM_RESERVE | MEM_COMMIT | MEM_LARGE_PAGES;
char * res = (char *)VirtualAlloc(NULL, char * res = (char *)VirtualAlloc(NULL, bytes, flag, PAGE_READWRITE);
bytes, if (res != NULL && exec) {
flag, DWORD oldprot;
PAGE_EXECUTE_READWRITE); // Windows doc says to use VirtualProtect to get execute permissions
VirtualProtect(res, bytes, PAGE_EXECUTE_READWRITE, &oldprot);
}
return res; return res;
} }
} }
...@@ -2703,7 +2711,7 @@ bool os::release_memory_special(char* base, size_t bytes) { ...@@ -2703,7 +2711,7 @@ bool os::release_memory_special(char* base, size_t bytes) {
void os::print_statistics() { void os::print_statistics() {
} }
bool os::commit_memory(char* addr, size_t bytes) { bool os::commit_memory(char* addr, size_t bytes, bool exec) {
if (bytes == 0) { if (bytes == 0) {
// Don't bother the OS with noops. // Don't bother the OS with noops.
return true; return true;
...@@ -2712,11 +2720,19 @@ bool os::commit_memory(char* addr, size_t bytes) { ...@@ -2712,11 +2720,19 @@ bool os::commit_memory(char* addr, size_t bytes) {
assert(bytes % os::vm_page_size() == 0, "commit in page-sized chunks"); assert(bytes % os::vm_page_size() == 0, "commit in page-sized chunks");
// Don't attempt to print anything if the OS call fails. We're // Don't attempt to print anything if the OS call fails. We're
// probably low on resources, so the print itself may cause crashes. // probably low on resources, so the print itself may cause crashes.
return VirtualAlloc(addr, bytes, MEM_COMMIT, PAGE_EXECUTE_READWRITE) != NULL; bool result = VirtualAlloc(addr, bytes, MEM_COMMIT, PAGE_READWRITE) != 0;
if (result != NULL && exec) {
DWORD oldprot;
// Windows doc says to use VirtualProtect to get execute permissions
return VirtualProtect(addr, bytes, PAGE_EXECUTE_READWRITE, &oldprot) != 0;
} else {
return result;
}
} }
bool os::commit_memory(char* addr, size_t size, size_t alignment_hint) { bool os::commit_memory(char* addr, size_t size, size_t alignment_hint,
return commit_memory(addr, size); bool exec) {
return commit_memory(addr, size, exec);
} }
bool os::uncommit_memory(char* addr, size_t bytes) { bool os::uncommit_memory(char* addr, size_t bytes) {
...@@ -2750,7 +2766,7 @@ bool os::protect_memory(char* addr, size_t bytes, ProtType prot, ...@@ -2750,7 +2766,7 @@ bool os::protect_memory(char* addr, size_t bytes, ProtType prot,
// Strange enough, but on Win32 one can change protection only for committed // Strange enough, but on Win32 one can change protection only for committed
// memory, not a big deal anyway, as bytes less or equal than 64K // memory, not a big deal anyway, as bytes less or equal than 64K
if (!is_committed && !commit_memory(addr, bytes)) { if (!is_committed && !commit_memory(addr, bytes, prot == MEM_PROT_RWX)) {
fatal("cannot commit protection page"); fatal("cannot commit protection page");
} }
// One cannot use os::guard_memory() here, as on Win32 guard page // One cannot use os::guard_memory() here, as on Win32 guard page
...@@ -3248,10 +3264,10 @@ jint os::init_2(void) { ...@@ -3248,10 +3264,10 @@ jint os::init_2(void) {
#endif #endif
if (!UseMembar) { if (!UseMembar) {
address mem_serialize_page = (address)VirtualAlloc(NULL, os::vm_page_size(), MEM_RESERVE, PAGE_EXECUTE_READWRITE); address mem_serialize_page = (address)VirtualAlloc(NULL, os::vm_page_size(), MEM_RESERVE, PAGE_READWRITE);
guarantee( mem_serialize_page != NULL, "Reserve Failed for memory serialize page"); guarantee( mem_serialize_page != NULL, "Reserve Failed for memory serialize page");
return_page = (address)VirtualAlloc(mem_serialize_page, os::vm_page_size(), MEM_COMMIT, PAGE_EXECUTE_READWRITE); return_page = (address)VirtualAlloc(mem_serialize_page, os::vm_page_size(), MEM_COMMIT, PAGE_READWRITE);
guarantee( return_page != NULL, "Commit Failed for memory serialize page"); guarantee( return_page != NULL, "Commit Failed for memory serialize page");
os::set_memory_serialize_page( mem_serialize_page ); os::set_memory_serialize_page( mem_serialize_page );
......
...@@ -112,7 +112,7 @@ bool CodeHeap::reserve(size_t reserved_size, size_t committed_size, ...@@ -112,7 +112,7 @@ bool CodeHeap::reserve(size_t reserved_size, size_t committed_size,
const size_t rs_align = page_size == (size_t) os::vm_page_size() ? 0 : const size_t rs_align = page_size == (size_t) os::vm_page_size() ? 0 :
MAX2(page_size, granularity); MAX2(page_size, granularity);
ReservedSpace rs(r_size, rs_align, rs_align > 0); ReservedCodeSpace rs(r_size, rs_align, rs_align > 0);
os::trace_page_sizes("code heap", committed_size, reserved_size, page_size, os::trace_page_sizes("code heap", committed_size, reserved_size, page_size,
rs.base(), rs.size()); rs.base(), rs.size());
if (!_memory.initialize(rs, c_size)) { if (!_memory.initialize(rs, c_size)) {
......
...@@ -202,8 +202,10 @@ class os: AllStatic { ...@@ -202,8 +202,10 @@ class os: AllStatic {
static char* attempt_reserve_memory_at(size_t bytes, char* addr); static char* attempt_reserve_memory_at(size_t bytes, char* addr);
static void split_reserved_memory(char *base, size_t size, static void split_reserved_memory(char *base, size_t size,
size_t split, bool realloc); size_t split, bool realloc);
static bool commit_memory(char* addr, size_t bytes); static bool commit_memory(char* addr, size_t bytes,
static bool commit_memory(char* addr, size_t size, size_t alignment_hint); bool executable = false);
static bool commit_memory(char* addr, size_t size, size_t alignment_hint,
bool executable = false);
static bool uncommit_memory(char* addr, size_t bytes); static bool uncommit_memory(char* addr, size_t bytes);
static bool release_memory(char* addr, size_t bytes); static bool release_memory(char* addr, size_t bytes);
...@@ -243,7 +245,8 @@ class os: AllStatic { ...@@ -243,7 +245,8 @@ class os: AllStatic {
static char* non_memory_address_word(); static char* non_memory_address_word();
// reserve, commit and pin the entire memory region // reserve, commit and pin the entire memory region
static char* reserve_memory_special(size_t size, char* addr = NULL); static char* reserve_memory_special(size_t size, char* addr = NULL,
bool executable = false);
static bool release_memory_special(char* addr, size_t bytes); static bool release_memory_special(char* addr, size_t bytes);
static bool large_page_init(); static bool large_page_init();
static size_t large_page_size(); static size_t large_page_size();
......
...@@ -28,7 +28,7 @@ ...@@ -28,7 +28,7 @@
// ReservedSpace // ReservedSpace
ReservedSpace::ReservedSpace(size_t size) { ReservedSpace::ReservedSpace(size_t size) {
initialize(size, 0, false, NULL, 0); initialize(size, 0, false, NULL, 0, false);
} }
ReservedSpace::ReservedSpace(size_t size, size_t alignment, ReservedSpace::ReservedSpace(size_t size, size_t alignment,
...@@ -36,7 +36,13 @@ ReservedSpace::ReservedSpace(size_t size, size_t alignment, ...@@ -36,7 +36,13 @@ ReservedSpace::ReservedSpace(size_t size, size_t alignment,
char* requested_address, char* requested_address,
const size_t noaccess_prefix) { const size_t noaccess_prefix) {
initialize(size+noaccess_prefix, alignment, large, requested_address, initialize(size+noaccess_prefix, alignment, large, requested_address,
noaccess_prefix); noaccess_prefix, false);
}
ReservedSpace::ReservedSpace(size_t size, size_t alignment,
bool large,
bool executable) {
initialize(size, alignment, large, NULL, 0, executable);
} }
char * char *
...@@ -132,7 +138,8 @@ ReservedSpace::ReservedSpace(const size_t prefix_size, ...@@ -132,7 +138,8 @@ ReservedSpace::ReservedSpace(const size_t prefix_size,
const bool try_reserve_special = UseLargePages && const bool try_reserve_special = UseLargePages &&
prefix_align == os::large_page_size(); prefix_align == os::large_page_size();
if (!os::can_commit_large_page_memory() && try_reserve_special) { if (!os::can_commit_large_page_memory() && try_reserve_special) {
initialize(size, prefix_align, true, requested_address, noaccess_prefix); initialize(size, prefix_align, true, requested_address, noaccess_prefix,
false);
return; return;
} }
...@@ -141,6 +148,7 @@ ReservedSpace::ReservedSpace(const size_t prefix_size, ...@@ -141,6 +148,7 @@ ReservedSpace::ReservedSpace(const size_t prefix_size,
_alignment = 0; _alignment = 0;
_special = false; _special = false;
_noaccess_prefix = 0; _noaccess_prefix = 0;
_executable = false;
// Assert that if noaccess_prefix is used, it is the same as prefix_align. // Assert that if noaccess_prefix is used, it is the same as prefix_align.
assert(noaccess_prefix == 0 || assert(noaccess_prefix == 0 ||
...@@ -189,7 +197,8 @@ ReservedSpace::ReservedSpace(const size_t prefix_size, ...@@ -189,7 +197,8 @@ ReservedSpace::ReservedSpace(const size_t prefix_size,
void ReservedSpace::initialize(size_t size, size_t alignment, bool large, void ReservedSpace::initialize(size_t size, size_t alignment, bool large,
char* requested_address, char* requested_address,
const size_t noaccess_prefix) { const size_t noaccess_prefix,
bool executable) {
const size_t granularity = os::vm_allocation_granularity(); const size_t granularity = os::vm_allocation_granularity();
assert((size & granularity - 1) == 0, assert((size & granularity - 1) == 0,
"size not aligned to os::vm_allocation_granularity()"); "size not aligned to os::vm_allocation_granularity()");
...@@ -201,6 +210,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large, ...@@ -201,6 +210,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large,
_base = NULL; _base = NULL;
_size = 0; _size = 0;
_special = false; _special = false;
_executable = executable;
_alignment = 0; _alignment = 0;
_noaccess_prefix = 0; _noaccess_prefix = 0;
if (size == 0) { if (size == 0) {
...@@ -214,7 +224,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large, ...@@ -214,7 +224,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large,
if (special) { if (special) {
base = os::reserve_memory_special(size, requested_address); base = os::reserve_memory_special(size, requested_address, executable);
if (base != NULL) { if (base != NULL) {
// Check alignment constraints // Check alignment constraints
...@@ -284,7 +294,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large, ...@@ -284,7 +294,7 @@ void ReservedSpace::initialize(size_t size, size_t alignment, bool large,
ReservedSpace::ReservedSpace(char* base, size_t size, size_t alignment, ReservedSpace::ReservedSpace(char* base, size_t size, size_t alignment,
bool special) { bool special, bool executable) {
assert((size % os::vm_allocation_granularity()) == 0, assert((size % os::vm_allocation_granularity()) == 0,
"size not allocation aligned"); "size not allocation aligned");
_base = base; _base = base;
...@@ -292,6 +302,7 @@ ReservedSpace::ReservedSpace(char* base, size_t size, size_t alignment, ...@@ -292,6 +302,7 @@ ReservedSpace::ReservedSpace(char* base, size_t size, size_t alignment,
_alignment = alignment; _alignment = alignment;
_noaccess_prefix = 0; _noaccess_prefix = 0;
_special = special; _special = special;
_executable = executable;
} }
...@@ -299,9 +310,10 @@ ReservedSpace ReservedSpace::first_part(size_t partition_size, size_t alignment, ...@@ -299,9 +310,10 @@ ReservedSpace ReservedSpace::first_part(size_t partition_size, size_t alignment,
bool split, bool realloc) { bool split, bool realloc) {
assert(partition_size <= size(), "partition failed"); assert(partition_size <= size(), "partition failed");
if (split) { if (split) {
os::split_reserved_memory(_base, _size, partition_size, realloc); os::split_reserved_memory(base(), size(), partition_size, realloc);
} }
ReservedSpace result(base(), partition_size, alignment, special()); ReservedSpace result(base(), partition_size, alignment, special(),
executable());
return result; return result;
} }
...@@ -310,7 +322,7 @@ ReservedSpace ...@@ -310,7 +322,7 @@ ReservedSpace
ReservedSpace::last_part(size_t partition_size, size_t alignment) { ReservedSpace::last_part(size_t partition_size, size_t alignment) {
assert(partition_size <= size(), "partition failed"); assert(partition_size <= size(), "partition failed");
ReservedSpace result(base() + partition_size, size() - partition_size, ReservedSpace result(base() + partition_size, size() - partition_size,
alignment, special()); alignment, special(), executable());
return result; return result;
} }
...@@ -348,6 +360,7 @@ void ReservedSpace::release() { ...@@ -348,6 +360,7 @@ void ReservedSpace::release() {
_size = 0; _size = 0;
_noaccess_prefix = 0; _noaccess_prefix = 0;
_special = false; _special = false;
_executable = false;
} }
} }
...@@ -396,6 +409,14 @@ ReservedHeapSpace::ReservedHeapSpace(const size_t prefix_size, ...@@ -396,6 +409,14 @@ ReservedHeapSpace::ReservedHeapSpace(const size_t prefix_size,
protect_noaccess_prefix(prefix_size+suffix_size); protect_noaccess_prefix(prefix_size+suffix_size);
} }
// Reserve space for code segment. Same as Java heap only we mark this as
// executable.
ReservedCodeSpace::ReservedCodeSpace(size_t r_size,
size_t rs_align,
bool large) :
ReservedSpace(r_size, rs_align, large, /*executable*/ true) {
}
// VirtualSpace // VirtualSpace
VirtualSpace::VirtualSpace() { VirtualSpace::VirtualSpace() {
...@@ -413,6 +434,7 @@ VirtualSpace::VirtualSpace() { ...@@ -413,6 +434,7 @@ VirtualSpace::VirtualSpace() {
_middle_alignment = 0; _middle_alignment = 0;
_upper_alignment = 0; _upper_alignment = 0;
_special = false; _special = false;
_executable = false;
} }
...@@ -426,6 +448,7 @@ bool VirtualSpace::initialize(ReservedSpace rs, size_t committed_size) { ...@@ -426,6 +448,7 @@ bool VirtualSpace::initialize(ReservedSpace rs, size_t committed_size) {
_high = low(); _high = low();
_special = rs.special(); _special = rs.special();
_executable = rs.executable();
// When a VirtualSpace begins life at a large size, make all future expansion // When a VirtualSpace begins life at a large size, make all future expansion
// and shrinking occur aligned to a granularity of large pages. This avoids // and shrinking occur aligned to a granularity of large pages. This avoids
...@@ -483,6 +506,7 @@ void VirtualSpace::release() { ...@@ -483,6 +506,7 @@ void VirtualSpace::release() {
_middle_alignment = 0; _middle_alignment = 0;
_upper_alignment = 0; _upper_alignment = 0;
_special = false; _special = false;
_executable = false;
} }
...@@ -592,7 +616,7 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) { ...@@ -592,7 +616,7 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) {
assert(low_boundary() <= lower_high() && assert(low_boundary() <= lower_high() &&
lower_high() + lower_needs <= lower_high_boundary(), lower_high() + lower_needs <= lower_high_boundary(),
"must not expand beyond region"); "must not expand beyond region");
if (!os::commit_memory(lower_high(), lower_needs)) { if (!os::commit_memory(lower_high(), lower_needs, _executable)) {
debug_only(warning("os::commit_memory failed")); debug_only(warning("os::commit_memory failed"));
return false; return false;
} else { } else {
...@@ -603,7 +627,8 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) { ...@@ -603,7 +627,8 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) {
assert(lower_high_boundary() <= middle_high() && assert(lower_high_boundary() <= middle_high() &&
middle_high() + middle_needs <= middle_high_boundary(), middle_high() + middle_needs <= middle_high_boundary(),
"must not expand beyond region"); "must not expand beyond region");
if (!os::commit_memory(middle_high(), middle_needs, middle_alignment())) { if (!os::commit_memory(middle_high(), middle_needs, middle_alignment(),
_executable)) {
debug_only(warning("os::commit_memory failed")); debug_only(warning("os::commit_memory failed"));
return false; return false;
} }
...@@ -613,7 +638,7 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) { ...@@ -613,7 +638,7 @@ bool VirtualSpace::expand_by(size_t bytes, bool pre_touch) {
assert(middle_high_boundary() <= upper_high() && assert(middle_high_boundary() <= upper_high() &&
upper_high() + upper_needs <= upper_high_boundary(), upper_high() + upper_needs <= upper_high_boundary(),
"must not expand beyond region"); "must not expand beyond region");
if (!os::commit_memory(upper_high(), upper_needs)) { if (!os::commit_memory(upper_high(), upper_needs, _executable)) {
debug_only(warning("os::commit_memory failed")); debug_only(warning("os::commit_memory failed"));
return false; return false;
} else { } else {
......
...@@ -32,12 +32,15 @@ class ReservedSpace VALUE_OBJ_CLASS_SPEC { ...@@ -32,12 +32,15 @@ class ReservedSpace VALUE_OBJ_CLASS_SPEC {
size_t _noaccess_prefix; size_t _noaccess_prefix;
size_t _alignment; size_t _alignment;
bool _special; bool _special;
bool _executable;
// ReservedSpace // ReservedSpace
ReservedSpace(char* base, size_t size, size_t alignment, bool special); ReservedSpace(char* base, size_t size, size_t alignment, bool special,
bool executable);
void initialize(size_t size, size_t alignment, bool large, void initialize(size_t size, size_t alignment, bool large,
char* requested_address, char* requested_address,
const size_t noaccess_prefix); const size_t noaccess_prefix,
bool executable);
// Release parts of an already-reserved memory region [addr, addr + len) to // Release parts of an already-reserved memory region [addr, addr + len) to
// get a new region that has "compound alignment." Return the start of the // get a new region that has "compound alignment." Return the start of the
...@@ -75,15 +78,15 @@ class ReservedSpace VALUE_OBJ_CLASS_SPEC { ...@@ -75,15 +78,15 @@ class ReservedSpace VALUE_OBJ_CLASS_SPEC {
const size_t suffix_size, const size_t suffix_align, const size_t suffix_size, const size_t suffix_align,
char* requested_address, char* requested_address,
const size_t noaccess_prefix = 0); const size_t noaccess_prefix = 0);
ReservedSpace(size_t size, size_t alignment, bool large, bool executable);
// Accessors // Accessors
char* base() const { return _base; } char* base() const { return _base; }
size_t size() const { return _size; } size_t size() const { return _size; }
size_t alignment() const { return _alignment; } size_t alignment() const { return _alignment; }
bool special() const { return _special; } bool special() const { return _special; }
bool executable() const { return _executable; }
size_t noaccess_prefix() const { return _noaccess_prefix; } size_t noaccess_prefix() const { return _noaccess_prefix; }
bool is_reserved() const { return _base != NULL; } bool is_reserved() const { return _base != NULL; }
void release(); void release();
...@@ -126,6 +129,13 @@ public: ...@@ -126,6 +129,13 @@ public:
char* requested_address); char* requested_address);
}; };
// Class encapsulating behavior specific memory space for Code
class ReservedCodeSpace : public ReservedSpace {
public:
// Constructor
ReservedCodeSpace(size_t r_size, size_t rs_align, bool large);
};
// VirtualSpace is data structure for committing a previously reserved address range in smaller chunks. // VirtualSpace is data structure for committing a previously reserved address range in smaller chunks.
class VirtualSpace VALUE_OBJ_CLASS_SPEC { class VirtualSpace VALUE_OBJ_CLASS_SPEC {
...@@ -143,6 +153,9 @@ class VirtualSpace VALUE_OBJ_CLASS_SPEC { ...@@ -143,6 +153,9 @@ class VirtualSpace VALUE_OBJ_CLASS_SPEC {
// os::commit_memory() or os::uncommit_memory(). // os::commit_memory() or os::uncommit_memory().
bool _special; bool _special;
// Need to know if commit should be executable.
bool _executable;
// MPSS Support // MPSS Support
// Each virtualspace region has a lower, middle, and upper region. // Each virtualspace region has a lower, middle, and upper region.
// Each region has an end boundary and a high pointer which is the // Each region has an end boundary and a high pointer which is the
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册