hotspot/src/os/solaris/vm/os_solaris.cpp
author tschatzl
Wed, 11 Sep 2013 16:25:02 +0200
changeset 19986 33d188c66ed9
parent 19546 f6b7c9e96ea3
child 22838 82c7497fbad4
child 22528 bd3821442010
permissions -rw-r--r--
8010722: assert: failed: heap size is too big for compressed oops Summary: Use conservative assumptions of required alignment for the various garbage collector components into account when determining the maximum heap size that supports compressed oops. Using this conservative value avoids several circular dependencies in the calculation. Reviewed-by: stefank, dholmes
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
     1
/*
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
     2
 * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
489c9b5090e2 Initial load
duke
parents:
diff changeset
     4
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
489c9b5090e2 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
489c9b5090e2 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
     8
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
489c9b5090e2 Initial load
duke
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
489c9b5090e2 Initial load
duke
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
489c9b5090e2 Initial load
duke
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
489c9b5090e2 Initial load
duke
parents:
diff changeset
    13
 * accompanied this code).
489c9b5090e2 Initial load
duke
parents:
diff changeset
    14
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
489c9b5090e2 Initial load
duke
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
489c9b5090e2 Initial load
duke
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
489c9b5090e2 Initial load
duke
parents:
diff changeset
    18
 *
5547
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 5403
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 5403
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 5403
diff changeset
    21
 * questions.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    22
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
    23
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
    24
7397
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    25
// no precompiled headers
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    26
#include "classfile/classLoader.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    27
#include "classfile/systemDictionary.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    28
#include "classfile/vmSymbols.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    29
#include "code/icBuffer.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    30
#include "code/vtableStubs.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    31
#include "compiler/compileBroker.hpp"
14626
0cf4eccf130f 8003240: x86: move MacroAssembler into separate file
twisti
parents: 14471
diff changeset
    32
#include "compiler/disassembler.hpp"
7397
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    33
#include "interpreter/interpreter.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    34
#include "jvm_solaris.h"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    35
#include "memory/allocation.inline.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    36
#include "memory/filemap.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    37
#include "mutex_solaris.inline.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    38
#include "oops/oop.inline.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    39
#include "os_share_solaris.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    40
#include "prims/jniFastGetField.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    41
#include "prims/jvm.h"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    42
#include "prims/jvm_misc.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    43
#include "runtime/arguments.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    44
#include "runtime/extendedPC.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    45
#include "runtime/globals.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    46
#include "runtime/interfaceSupport.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    47
#include "runtime/java.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    48
#include "runtime/javaCalls.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    49
#include "runtime/mutexLocker.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    50
#include "runtime/objectMonitor.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    51
#include "runtime/osThread.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    52
#include "runtime/perfMemory.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    53
#include "runtime/sharedRuntime.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    54
#include "runtime/statSampler.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    55
#include "runtime/stubRoutines.hpp"
14583
d70ee55535f4 8003935: Simplify the needed includes for using Thread::current()
stefank
parents: 14471
diff changeset
    56
#include "runtime/thread.inline.hpp"
7397
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    57
#include "runtime/threadCritical.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    58
#include "runtime/timer.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    59
#include "services/attachListener.hpp"
14120
7d298141c258 7199092: NMT: NMT needs to deal overlapped virtual memory ranges
zgu
parents: 13975
diff changeset
    60
#include "services/memTracker.hpp"
7397
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    61
#include "services/runtimeService.hpp"
7447
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
    62
#include "utilities/decoder.hpp"
7397
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    63
#include "utilities/defaultStream.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    64
#include "utilities/events.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    65
#include "utilities/growableArray.hpp"
5b173b4ca846 6989984: Use standard include model for Hospot
stefank
parents: 7392
diff changeset
    66
#include "utilities/vmError.hpp"
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    67
489c9b5090e2 Initial load
duke
parents:
diff changeset
    68
// put OS-includes here
489c9b5090e2 Initial load
duke
parents:
diff changeset
    69
# include <dlfcn.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    70
# include <errno.h>
7692
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
    71
# include <exception>
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    72
# include <link.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    73
# include <poll.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    74
# include <pthread.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    75
# include <pwd.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    76
# include <schedctl.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    77
# include <setjmp.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    78
# include <signal.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    79
# include <stdio.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    80
# include <alloca.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    81
# include <sys/filio.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    82
# include <sys/ipc.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    83
# include <sys/lwp.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    84
# include <sys/machelf.h>     // for elf Sym structure used by dladdr1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    85
# include <sys/mman.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    86
# include <sys/processor.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    87
# include <sys/procset.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    88
# include <sys/pset.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    89
# include <sys/resource.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    90
# include <sys/shm.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    91
# include <sys/socket.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    92
# include <sys/stat.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    93
# include <sys/systeminfo.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    94
# include <sys/time.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    95
# include <sys/times.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    96
# include <sys/types.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    97
# include <sys/wait.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    98
# include <sys/utsname.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
    99
# include <thread.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   100
# include <unistd.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   101
# include <sys/priocntl.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   102
# include <sys/rtpriocntl.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   103
# include <sys/tspriocntl.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   104
# include <sys/iapriocntl.h>
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   105
# include <sys/fxpriocntl.h>
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   106
# include <sys/loadavg.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   107
# include <string.h>
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
   108
# include <stdio.h>
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   109
489c9b5090e2 Initial load
duke
parents:
diff changeset
   110
# define _STRUCTURED_PROC 1  //  this gets us the new structured proc interfaces of 5.6 & later
489c9b5090e2 Initial load
duke
parents:
diff changeset
   111
# include <sys/procfs.h>     //  see comment in <sys/procfs.h>
489c9b5090e2 Initial load
duke
parents:
diff changeset
   112
489c9b5090e2 Initial load
duke
parents:
diff changeset
   113
#define MAX_PATH (2 * K)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   114
489c9b5090e2 Initial load
duke
parents:
diff changeset
   115
// for timer info max values which include all bits
489c9b5090e2 Initial load
duke
parents:
diff changeset
   116
#define ALL_64_BITS CONST64(0xFFFFFFFFFFFFFFFF)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   117
489c9b5090e2 Initial load
duke
parents:
diff changeset
   118
489c9b5090e2 Initial load
duke
parents:
diff changeset
   119
// Here are some liblgrp types from sys/lgrp_user.h to be able to
489c9b5090e2 Initial load
duke
parents:
diff changeset
   120
// compile on older systems without this header file.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   121
489c9b5090e2 Initial load
duke
parents:
diff changeset
   122
#ifndef MADV_ACCESS_LWP
489c9b5090e2 Initial load
duke
parents:
diff changeset
   123
# define  MADV_ACCESS_LWP         7       /* next LWP to access heavily */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   124
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
   125
#ifndef MADV_ACCESS_MANY
489c9b5090e2 Initial load
duke
parents:
diff changeset
   126
# define  MADV_ACCESS_MANY        8       /* many processes to access heavily */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   127
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
   128
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   129
#ifndef LGRP_RSRC_CPU
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   130
# define LGRP_RSRC_CPU           0       /* CPU resources */
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   131
#endif
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   132
#ifndef LGRP_RSRC_MEM
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   133
# define LGRP_RSRC_MEM           1       /* memory resources */
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   134
#endif
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
   135
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   136
// see thr_setprio(3T) for the basis of these numbers
489c9b5090e2 Initial load
duke
parents:
diff changeset
   137
#define MinimumPriority 0
489c9b5090e2 Initial load
duke
parents:
diff changeset
   138
#define NormalPriority  64
489c9b5090e2 Initial load
duke
parents:
diff changeset
   139
#define MaximumPriority 127
489c9b5090e2 Initial load
duke
parents:
diff changeset
   140
489c9b5090e2 Initial load
duke
parents:
diff changeset
   141
// Values for ThreadPriorityPolicy == 1
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   142
int prio_policy1[CriticalPriority+1] = {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   143
  -99999,  0, 16,  32,  48,  64,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   144
          80, 96, 112, 124, 127, 127 };
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   145
489c9b5090e2 Initial load
duke
parents:
diff changeset
   146
// System parameters used internally
489c9b5090e2 Initial load
duke
parents:
diff changeset
   147
static clock_t clock_tics_per_sec = 100;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   148
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
   149
// Track if we have called enable_extended_FILE_stdio (on Solaris 10u4+)
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
   150
static bool enabled_extended_FILE_stdio = false;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
   151
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   152
// For diagnostics to print a message once. see run_periodic_checks
489c9b5090e2 Initial load
duke
parents:
diff changeset
   153
static bool check_addr0_done = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   154
static sigset_t check_signal_done;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   155
static bool check_signals = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   156
489c9b5090e2 Initial load
duke
parents:
diff changeset
   157
address os::Solaris::handler_start;  // start pc of thr_sighndlrinfo
489c9b5090e2 Initial load
duke
parents:
diff changeset
   158
address os::Solaris::handler_end;    // end pc of thr_sighndlrinfo
489c9b5090e2 Initial load
duke
parents:
diff changeset
   159
489c9b5090e2 Initial load
duke
parents:
diff changeset
   160
address os::Solaris::_main_stack_base = NULL;  // 4352906 workaround
489c9b5090e2 Initial load
duke
parents:
diff changeset
   161
489c9b5090e2 Initial load
duke
parents:
diff changeset
   162
489c9b5090e2 Initial load
duke
parents:
diff changeset
   163
// "default" initializers for missing libc APIs
489c9b5090e2 Initial load
duke
parents:
diff changeset
   164
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   165
  static int lwp_mutex_init(mutex_t *mx, int scope, void *arg) { memset(mx, 0, sizeof(mutex_t)); return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   166
  static int lwp_mutex_destroy(mutex_t *mx)                 { return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   167
489c9b5090e2 Initial load
duke
parents:
diff changeset
   168
  static int lwp_cond_init(cond_t *cv, int scope, void *arg){ memset(cv, 0, sizeof(cond_t)); return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   169
  static int lwp_cond_destroy(cond_t *cv)                   { return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   170
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   171
489c9b5090e2 Initial load
duke
parents:
diff changeset
   172
// "default" initializers for pthread-based synchronization
489c9b5090e2 Initial load
duke
parents:
diff changeset
   173
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   174
  static int pthread_mutex_default_init(mutex_t *mx, int scope, void *arg) { memset(mx, 0, sizeof(mutex_t)); return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   175
  static int pthread_cond_default_init(cond_t *cv, int scope, void *arg){ memset(cv, 0, sizeof(cond_t)); return 0; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   176
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   177
18025
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
   178
static void unpackTime(timespec* absTime, bool isAbsolute, jlong time);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
   179
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   180
// Thread Local Storage
489c9b5090e2 Initial load
duke
parents:
diff changeset
   181
// This is common to all Solaris platforms so it is defined here,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   182
// in this common file.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   183
// The declarations are in the os_cpu threadLS*.hpp files.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   184
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
   185
// Static member initialization for TLS
489c9b5090e2 Initial load
duke
parents:
diff changeset
   186
Thread* ThreadLocalStorage::_get_thread_cache[ThreadLocalStorage::_pd_cache_size] = {NULL};
489c9b5090e2 Initial load
duke
parents:
diff changeset
   187
489c9b5090e2 Initial load
duke
parents:
diff changeset
   188
#ifndef PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
   189
#define _PCT(n,d)       ((100.0*(double)(n))/(double)(d))
489c9b5090e2 Initial load
duke
parents:
diff changeset
   190
489c9b5090e2 Initial load
duke
parents:
diff changeset
   191
int ThreadLocalStorage::_tcacheHit = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   192
int ThreadLocalStorage::_tcacheMiss = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   193
489c9b5090e2 Initial load
duke
parents:
diff changeset
   194
void ThreadLocalStorage::print_statistics() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   195
  int total = _tcacheMiss+_tcacheHit;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   196
  tty->print_cr("Thread cache hits %d misses %d total %d percent %f\n",
489c9b5090e2 Initial load
duke
parents:
diff changeset
   197
                _tcacheHit, _tcacheMiss, total, _PCT(_tcacheHit, total));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   198
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   199
#undef _PCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
   200
#endif // PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
   201
489c9b5090e2 Initial load
duke
parents:
diff changeset
   202
Thread* ThreadLocalStorage::get_thread_via_cache_slowly(uintptr_t raw_id,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   203
                                                        int index) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   204
  Thread *thread = get_thread_slow();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   205
  if (thread != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   206
    address sp = os::current_stack_pointer();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   207
    guarantee(thread->_stack_base == NULL ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
   208
              (sp <= thread->_stack_base &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
   209
                 sp >= thread->_stack_base - thread->_stack_size) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
   210
               is_error_reported(),
489c9b5090e2 Initial load
duke
parents:
diff changeset
   211
              "sp must be inside of selected thread stack");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   212
7724
a92d706dbdd5 7003271: Hotspot should track cumulative Java heap bytes allocated on a per-thread basis
phh
parents: 7721
diff changeset
   213
    thread->set_self_raw_id(raw_id);  // mark for quick retrieval
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   214
    _get_thread_cache[ index ] = thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   215
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   216
  return thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   217
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   218
489c9b5090e2 Initial load
duke
parents:
diff changeset
   219
489c9b5090e2 Initial load
duke
parents:
diff changeset
   220
static const double all_zero[ sizeof(Thread) / sizeof(double) + 1 ] = {0};
489c9b5090e2 Initial load
duke
parents:
diff changeset
   221
#define NO_CACHED_THREAD ((Thread*)all_zero)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   222
489c9b5090e2 Initial load
duke
parents:
diff changeset
   223
void ThreadLocalStorage::pd_set_thread(Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   224
489c9b5090e2 Initial load
duke
parents:
diff changeset
   225
  // Store the new value before updating the cache to prevent a race
489c9b5090e2 Initial load
duke
parents:
diff changeset
   226
  // between get_thread_via_cache_slowly() and this store operation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   227
  os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   228
489c9b5090e2 Initial load
duke
parents:
diff changeset
   229
  // Update thread cache with new thread if setting on thread create,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   230
  // or NO_CACHED_THREAD (zeroed) thread if resetting thread on exit.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   231
  uintptr_t raw = pd_raw_thread_id();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   232
  int ix = pd_cache_index(raw);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   233
  _get_thread_cache[ix] = thread == NULL ? NO_CACHED_THREAD : thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   234
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   235
489c9b5090e2 Initial load
duke
parents:
diff changeset
   236
void ThreadLocalStorage::pd_init() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   237
  for (int i = 0; i < _pd_cache_size; i++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   238
    _get_thread_cache[i] = NO_CACHED_THREAD;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   239
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   240
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   241
489c9b5090e2 Initial load
duke
parents:
diff changeset
   242
// Invalidate all the caches (happens to be the same as pd_init).
489c9b5090e2 Initial load
duke
parents:
diff changeset
   243
void ThreadLocalStorage::pd_invalidate_all() { pd_init(); }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   244
489c9b5090e2 Initial load
duke
parents:
diff changeset
   245
#undef NO_CACHED_THREAD
489c9b5090e2 Initial load
duke
parents:
diff changeset
   246
489c9b5090e2 Initial load
duke
parents:
diff changeset
   247
// END Thread Local Storage
489c9b5090e2 Initial load
duke
parents:
diff changeset
   248
489c9b5090e2 Initial load
duke
parents:
diff changeset
   249
static inline size_t adjust_stack_size(address base, size_t size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   250
  if ((ssize_t)size < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   251
    // 4759953: Compensate for ridiculous stack size.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   252
    size = max_intx;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   253
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   254
  if (size > (size_t)base) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   255
    // 4812466: Make sure size doesn't allow the stack to wrap the address space.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   256
    size = (size_t)base;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   257
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   258
  return size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   259
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   260
489c9b5090e2 Initial load
duke
parents:
diff changeset
   261
static inline stack_t get_stack_info() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   262
  stack_t st;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   263
  int retval = thr_stksegment(&st);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   264
  st.ss_size = adjust_stack_size((address)st.ss_sp, st.ss_size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   265
  assert(retval == 0, "incorrect return value from thr_stksegment");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   266
  assert((address)&st < (address)st.ss_sp, "Invalid stack base returned");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   267
  assert((address)&st > (address)st.ss_sp-st.ss_size, "Invalid stack size returned");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   268
  return st;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   269
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   270
489c9b5090e2 Initial load
duke
parents:
diff changeset
   271
address os::current_stack_base() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   272
  int r = thr_main() ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   273
  guarantee (r == 0 || r == 1, "CR6501650 or CR6493689") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   274
  bool is_primordial_thread = r;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   275
489c9b5090e2 Initial load
duke
parents:
diff changeset
   276
  // Workaround 4352906, avoid calls to thr_stksegment by
489c9b5090e2 Initial load
duke
parents:
diff changeset
   277
  // thr_main after the first one (it looks like we trash
489c9b5090e2 Initial load
duke
parents:
diff changeset
   278
  // some data, causing the value for ss_sp to be incorrect).
489c9b5090e2 Initial load
duke
parents:
diff changeset
   279
  if (!is_primordial_thread || os::Solaris::_main_stack_base == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   280
    stack_t st = get_stack_info();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   281
    if (is_primordial_thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   282
      // cache initial value of stack base
489c9b5090e2 Initial load
duke
parents:
diff changeset
   283
      os::Solaris::_main_stack_base = (address)st.ss_sp;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   284
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   285
    return (address)st.ss_sp;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   286
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   287
    guarantee(os::Solaris::_main_stack_base != NULL, "Attempt to use null cached stack base");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   288
    return os::Solaris::_main_stack_base;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   289
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   290
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   291
489c9b5090e2 Initial load
duke
parents:
diff changeset
   292
size_t os::current_stack_size() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   293
  size_t size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   294
489c9b5090e2 Initial load
duke
parents:
diff changeset
   295
  int r = thr_main() ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   296
  guarantee (r == 0 || r == 1, "CR6501650 or CR6493689") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   297
  if(!r) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   298
    size = get_stack_info().ss_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   299
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   300
    struct rlimit limits;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   301
    getrlimit(RLIMIT_STACK, &limits);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   302
    size = adjust_stack_size(os::Solaris::_main_stack_base, (size_t)limits.rlim_cur);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   303
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   304
  // base may not be page aligned
489c9b5090e2 Initial load
duke
parents:
diff changeset
   305
  address base = current_stack_base();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   306
  address bottom = (address)align_size_up((intptr_t)(base - size), os::vm_page_size());;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   307
  return (size_t)(base - bottom);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   308
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   309
2012
041fbc6030dd 6800586: -XX:+PrintGCDateStamps is using mt-unsafe localtime function
ysr
parents: 1686
diff changeset
   310
struct tm* os::localtime_pd(const time_t* clock, struct tm*  res) {
041fbc6030dd 6800586: -XX:+PrintGCDateStamps is using mt-unsafe localtime function
ysr
parents: 1686
diff changeset
   311
  return localtime_r(clock, res);
041fbc6030dd 6800586: -XX:+PrintGCDateStamps is using mt-unsafe localtime function
ysr
parents: 1686
diff changeset
   312
}
041fbc6030dd 6800586: -XX:+PrintGCDateStamps is using mt-unsafe localtime function
ysr
parents: 1686
diff changeset
   313
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   314
// interruptible infrastructure
489c9b5090e2 Initial load
duke
parents:
diff changeset
   315
489c9b5090e2 Initial load
duke
parents:
diff changeset
   316
// setup_interruptible saves the thread state before going into an
489c9b5090e2 Initial load
duke
parents:
diff changeset
   317
// interruptible system call.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   318
// The saved state is used to restore the thread to
489c9b5090e2 Initial load
duke
parents:
diff changeset
   319
// its former state whether or not an interrupt is received.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   320
// Used by classloader os::read
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
   321
// os::restartable_read calls skip this layer and stay in _thread_in_native
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   322
489c9b5090e2 Initial load
duke
parents:
diff changeset
   323
void os::Solaris::setup_interruptible(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   324
489c9b5090e2 Initial load
duke
parents:
diff changeset
   325
  JavaThreadState thread_state = thread->thread_state();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   326
489c9b5090e2 Initial load
duke
parents:
diff changeset
   327
  assert(thread_state != _thread_blocked, "Coming from the wrong thread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   328
  assert(thread_state != _thread_in_native, "Native threads skip setup_interruptible");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   329
  OSThread* osthread = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   330
  osthread->set_saved_interrupt_thread_state(thread_state);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   331
  thread->frame_anchor()->make_walkable(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   332
  ThreadStateTransition::transition(thread, thread_state, _thread_blocked);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   333
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   334
489c9b5090e2 Initial load
duke
parents:
diff changeset
   335
// Version of setup_interruptible() for threads that are already in
489c9b5090e2 Initial load
duke
parents:
diff changeset
   336
// _thread_blocked. Used by os_sleep().
489c9b5090e2 Initial load
duke
parents:
diff changeset
   337
void os::Solaris::setup_interruptible_already_blocked(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   338
  thread->frame_anchor()->make_walkable(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   339
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   340
489c9b5090e2 Initial load
duke
parents:
diff changeset
   341
JavaThread* os::Solaris::setup_interruptible() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   342
  JavaThread* thread = (JavaThread*)ThreadLocalStorage::thread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   343
  setup_interruptible(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   344
  return thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   345
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   346
489c9b5090e2 Initial load
duke
parents:
diff changeset
   347
void os::Solaris::try_enable_extended_io() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   348
  typedef int (*enable_extended_FILE_stdio_t)(int, int);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   349
489c9b5090e2 Initial load
duke
parents:
diff changeset
   350
  if (!UseExtendedFileIO) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   351
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   352
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   353
489c9b5090e2 Initial load
duke
parents:
diff changeset
   354
  enable_extended_FILE_stdio_t enabler =
489c9b5090e2 Initial load
duke
parents:
diff changeset
   355
    (enable_extended_FILE_stdio_t) dlsym(RTLD_DEFAULT,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   356
                                         "enable_extended_FILE_stdio");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   357
  if (enabler) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   358
    enabler(-1, -1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   359
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   360
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   361
489c9b5090e2 Initial load
duke
parents:
diff changeset
   362
489c9b5090e2 Initial load
duke
parents:
diff changeset
   363
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
   364
489c9b5090e2 Initial load
duke
parents:
diff changeset
   365
JavaThread* os::Solaris::setup_interruptible_native() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   366
  JavaThread* thread = (JavaThread*)ThreadLocalStorage::thread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   367
  JavaThreadState thread_state = thread->thread_state();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   368
  assert(thread_state == _thread_in_native, "Assumed thread_in_native");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   369
  return thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   370
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   371
489c9b5090e2 Initial load
duke
parents:
diff changeset
   372
void os::Solaris::cleanup_interruptible_native(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   373
  JavaThreadState thread_state = thread->thread_state();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   374
  assert(thread_state == _thread_in_native, "Assumed thread_in_native");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   375
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   376
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
   377
489c9b5090e2 Initial load
duke
parents:
diff changeset
   378
// cleanup_interruptible reverses the effects of setup_interruptible
489c9b5090e2 Initial load
duke
parents:
diff changeset
   379
// setup_interruptible_already_blocked() does not need any cleanup.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   380
489c9b5090e2 Initial load
duke
parents:
diff changeset
   381
void os::Solaris::cleanup_interruptible(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   382
  OSThread* osthread = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   383
489c9b5090e2 Initial load
duke
parents:
diff changeset
   384
  ThreadStateTransition::transition(thread, _thread_blocked, osthread->saved_interrupt_thread_state());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   385
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   386
489c9b5090e2 Initial load
duke
parents:
diff changeset
   387
// I/O interruption related counters called in _INTERRUPTIBLE
489c9b5090e2 Initial load
duke
parents:
diff changeset
   388
489c9b5090e2 Initial load
duke
parents:
diff changeset
   389
void os::Solaris::bump_interrupted_before_count() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   390
  RuntimeService::record_interrupted_before_count();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   391
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   392
489c9b5090e2 Initial load
duke
parents:
diff changeset
   393
void os::Solaris::bump_interrupted_during_count() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   394
  RuntimeService::record_interrupted_during_count();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   395
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   396
489c9b5090e2 Initial load
duke
parents:
diff changeset
   397
static int _processors_online = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   398
489c9b5090e2 Initial load
duke
parents:
diff changeset
   399
         jint os::Solaris::_os_thread_limit = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   400
volatile jint os::Solaris::_os_thread_count = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   401
489c9b5090e2 Initial load
duke
parents:
diff changeset
   402
julong os::available_memory() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   403
  return Solaris::available_memory();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   404
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   405
489c9b5090e2 Initial load
duke
parents:
diff changeset
   406
julong os::Solaris::available_memory() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   407
  return (julong)sysconf(_SC_AVPHYS_PAGES) * os::vm_page_size();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   408
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   409
489c9b5090e2 Initial load
duke
parents:
diff changeset
   410
julong os::Solaris::_physical_memory = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   411
489c9b5090e2 Initial load
duke
parents:
diff changeset
   412
julong os::physical_memory() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   413
   return Solaris::physical_memory();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   414
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   415
489c9b5090e2 Initial load
duke
parents:
diff changeset
   416
static hrtime_t first_hrtime = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   417
static const hrtime_t hrtime_hz = 1000*1000*1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   418
const int LOCK_BUSY = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   419
const int LOCK_FREE = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   420
const int LOCK_INVALID = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   421
static volatile hrtime_t max_hrtime = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   422
static volatile int max_hrtime_lock = LOCK_FREE;     // Update counter with LSB as lock-in-progress
489c9b5090e2 Initial load
duke
parents:
diff changeset
   423
489c9b5090e2 Initial load
duke
parents:
diff changeset
   424
489c9b5090e2 Initial load
duke
parents:
diff changeset
   425
void os::Solaris::initialize_system_info() {
4493
9204129f065e 6843629: Make current hotspot build part of jdk5 control build
phh
parents: 4487
diff changeset
   426
  set_processor_count(sysconf(_SC_NPROCESSORS_CONF));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   427
  _processors_online = sysconf (_SC_NPROCESSORS_ONLN);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   428
  _physical_memory = (julong)sysconf(_SC_PHYS_PAGES) * (julong)sysconf(_SC_PAGESIZE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   429
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   430
489c9b5090e2 Initial load
duke
parents:
diff changeset
   431
int os::active_processor_count() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   432
  int online_cpus = sysconf(_SC_NPROCESSORS_ONLN);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   433
  pid_t pid = getpid();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   434
  psetid_t pset = PS_NONE;
1419
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   435
  // Are we running in a processor set or is there any processor set around?
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   436
  if (pset_bind(PS_QUERY, P_PID, pid, &pset) == 0) {
1419
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   437
    uint_t pset_cpus;
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   438
    // Query the number of cpus available to us.
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   439
    if (pset_info(pset, NULL, &pset_cpus, NULL) == 0) {
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   440
      assert(pset_cpus > 0 && pset_cpus <= online_cpus, "sanity check");
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   441
      _processors_online = pset_cpus;
561c0debbf22 6673124: Runtime.availableProcessors / os::active_processor_count wrong if unused processor sets exist
xlu
parents: 1388
diff changeset
   442
      return pset_cpus;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   443
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   444
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   445
  // Otherwise return number of online cpus
489c9b5090e2 Initial load
duke
parents:
diff changeset
   446
  return online_cpus;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   447
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   448
489c9b5090e2 Initial load
duke
parents:
diff changeset
   449
static bool find_processors_in_pset(psetid_t        pset,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   450
                                    processorid_t** id_array,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   451
                                    uint_t*         id_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   452
  bool result = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   453
  // Find the number of processors in the processor set.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   454
  if (pset_info(pset, NULL, id_length, NULL) == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   455
    // Make up an array to hold their ids.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   456
    *id_array = NEW_C_HEAP_ARRAY(processorid_t, *id_length, mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   457
    // Fill in the array with their processor ids.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   458
    if (pset_info(pset, NULL, id_length, *id_array) == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   459
      result = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   460
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   461
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   462
  return result;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   463
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   464
489c9b5090e2 Initial load
duke
parents:
diff changeset
   465
// Callers of find_processors_online() must tolerate imprecise results --
489c9b5090e2 Initial load
duke
parents:
diff changeset
   466
// the system configuration can change asynchronously because of DR
489c9b5090e2 Initial load
duke
parents:
diff changeset
   467
// or explicit psradm operations.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   468
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
   469
// We also need to take care that the loop (below) terminates as the
489c9b5090e2 Initial load
duke
parents:
diff changeset
   470
// number of processors online can change between the _SC_NPROCESSORS_ONLN
489c9b5090e2 Initial load
duke
parents:
diff changeset
   471
// request and the loop that builds the list of processor ids.   Unfortunately
489c9b5090e2 Initial load
duke
parents:
diff changeset
   472
// there's no reliable way to determine the maximum valid processor id,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   473
// so we use a manifest constant, MAX_PROCESSOR_ID, instead.  See p_online
489c9b5090e2 Initial load
duke
parents:
diff changeset
   474
// man pages, which claim the processor id set is "sparse, but
489c9b5090e2 Initial load
duke
parents:
diff changeset
   475
// not too sparse".  MAX_PROCESSOR_ID is used to ensure that we eventually
489c9b5090e2 Initial load
duke
parents:
diff changeset
   476
// exit the loop.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   477
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
   478
// In the future we'll be able to use sysconf(_SC_CPUID_MAX), but that's
489c9b5090e2 Initial load
duke
parents:
diff changeset
   479
// not available on S8.0.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   480
489c9b5090e2 Initial load
duke
parents:
diff changeset
   481
static bool find_processors_online(processorid_t** id_array,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   482
                                   uint*           id_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   483
  const processorid_t MAX_PROCESSOR_ID = 100000 ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   484
  // Find the number of processors online.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   485
  *id_length = sysconf(_SC_NPROCESSORS_ONLN);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   486
  // Make up an array to hold their ids.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   487
  *id_array = NEW_C_HEAP_ARRAY(processorid_t, *id_length, mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   488
  // Processors need not be numbered consecutively.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   489
  long found = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   490
  processorid_t next = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   491
  while (found < *id_length && next < MAX_PROCESSOR_ID) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   492
    processor_info_t info;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   493
    if (processor_info(next, &info) == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   494
      // NB, PI_NOINTR processors are effectively online ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
   495
      if (info.pi_state == P_ONLINE || info.pi_state == P_NOINTR) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   496
        (*id_array)[found] = next;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   497
        found += 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   498
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   499
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   500
    next += 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   501
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   502
  if (found < *id_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   503
      // The loop above didn't identify the expected number of processors.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   504
      // We could always retry the operation, calling sysconf(_SC_NPROCESSORS_ONLN)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   505
      // and re-running the loop, above, but there's no guarantee of progress
489c9b5090e2 Initial load
duke
parents:
diff changeset
   506
      // if the system configuration is in flux.  Instead, we just return what
489c9b5090e2 Initial load
duke
parents:
diff changeset
   507
      // we've got.  Note that in the worst case find_processors_online() could
489c9b5090e2 Initial load
duke
parents:
diff changeset
   508
      // return an empty set.  (As a fall-back in the case of the empty set we
489c9b5090e2 Initial load
duke
parents:
diff changeset
   509
      // could just return the ID of the current processor).
489c9b5090e2 Initial load
duke
parents:
diff changeset
   510
      *id_length = found ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   511
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   512
489c9b5090e2 Initial load
duke
parents:
diff changeset
   513
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   514
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   515
489c9b5090e2 Initial load
duke
parents:
diff changeset
   516
static bool assign_distribution(processorid_t* id_array,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   517
                                uint           id_length,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   518
                                uint*          distribution,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   519
                                uint           distribution_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   520
  // We assume we can assign processorid_t's to uint's.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   521
  assert(sizeof(processorid_t) == sizeof(uint),
489c9b5090e2 Initial load
duke
parents:
diff changeset
   522
         "can't convert processorid_t to uint");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   523
  // Quick check to see if we won't succeed.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   524
  if (id_length < distribution_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   525
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   526
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   527
  // Assign processor ids to the distribution.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   528
  // Try to shuffle processors to distribute work across boards,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   529
  // assuming 4 processors per board.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   530
  const uint processors_per_board = ProcessDistributionStride;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   531
  // Find the maximum processor id.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   532
  processorid_t max_id = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   533
  for (uint m = 0; m < id_length; m += 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   534
    max_id = MAX2(max_id, id_array[m]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   535
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   536
  // The next id, to limit loops.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   537
  const processorid_t limit_id = max_id + 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   538
  // Make up markers for available processors.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   539
  bool* available_id = NEW_C_HEAP_ARRAY(bool, limit_id, mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   540
  for (uint c = 0; c < limit_id; c += 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   541
    available_id[c] = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   542
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   543
  for (uint a = 0; a < id_length; a += 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   544
    available_id[id_array[a]] = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   545
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   546
  // Step by "boards", then by "slot", copying to "assigned".
489c9b5090e2 Initial load
duke
parents:
diff changeset
   547
  // NEEDS_CLEANUP: The assignment of processors should be stateful,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   548
  //                remembering which processors have been assigned by
489c9b5090e2 Initial load
duke
parents:
diff changeset
   549
  //                previous calls, etc., so as to distribute several
489c9b5090e2 Initial load
duke
parents:
diff changeset
   550
  //                independent calls of this method.  What we'd like is
489c9b5090e2 Initial load
duke
parents:
diff changeset
   551
  //                It would be nice to have an API that let us ask
489c9b5090e2 Initial load
duke
parents:
diff changeset
   552
  //                how many processes are bound to a processor,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   553
  //                but we don't have that, either.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   554
  //                In the short term, "board" is static so that
489c9b5090e2 Initial load
duke
parents:
diff changeset
   555
  //                subsequent distributions don't all start at board 0.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   556
  static uint board = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   557
  uint assigned = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   558
  // Until we've found enough processors ....
489c9b5090e2 Initial load
duke
parents:
diff changeset
   559
  while (assigned < distribution_length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   560
    // ... find the next available processor in the board.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   561
    for (uint slot = 0; slot < processors_per_board; slot += 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   562
      uint try_id = board * processors_per_board + slot;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   563
      if ((try_id < limit_id) && (available_id[try_id] == true)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   564
        distribution[assigned] = try_id;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   565
        available_id[try_id] = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   566
        assigned += 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   567
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   568
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   569
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   570
    board += 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   571
    if (board * processors_per_board + 0 >= limit_id) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   572
      board = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   573
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   574
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   575
  if (available_id != NULL) {
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   576
    FREE_C_HEAP_ARRAY(bool, available_id, mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   577
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   578
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   579
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   580
10739
91935236600e 7098194: integrate macosx-port changes
dcubed
parents: 10561
diff changeset
   581
void os::set_native_thread_name(const char *name) {
91935236600e 7098194: integrate macosx-port changes
dcubed
parents: 10561
diff changeset
   582
  // Not yet implemented.
91935236600e 7098194: integrate macosx-port changes
dcubed
parents: 10561
diff changeset
   583
  return;
91935236600e 7098194: integrate macosx-port changes
dcubed
parents: 10561
diff changeset
   584
}
91935236600e 7098194: integrate macosx-port changes
dcubed
parents: 10561
diff changeset
   585
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   586
bool os::distribute_processes(uint length, uint* distribution) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   587
  bool result = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   588
  // Find the processor id's of all the available CPUs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   589
  processorid_t* id_array  = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   590
  uint           id_length = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   591
  // There are some races between querying information and using it,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   592
  // since processor sets can change dynamically.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   593
  psetid_t pset = PS_NONE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   594
  // Are we running in a processor set?
489c9b5090e2 Initial load
duke
parents:
diff changeset
   595
  if ((pset_bind(PS_QUERY, P_PID, P_MYID, &pset) == 0) && pset != PS_NONE) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   596
    result = find_processors_in_pset(pset, &id_array, &id_length);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   597
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   598
    result = find_processors_online(&id_array, &id_length);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   599
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   600
  if (result == true) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   601
    if (id_length >= length) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   602
      result = assign_distribution(id_array, id_length, distribution, length);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   603
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   604
      result = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   605
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   606
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   607
  if (id_array != NULL) {
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   608
    FREE_C_HEAP_ARRAY(processorid_t, id_array, mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   609
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   610
  return result;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   611
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   612
489c9b5090e2 Initial load
duke
parents:
diff changeset
   613
bool os::bind_to_processor(uint processor_id) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   614
  // We assume that a processorid_t can be stored in a uint.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   615
  assert(sizeof(uint) == sizeof(processorid_t),
489c9b5090e2 Initial load
duke
parents:
diff changeset
   616
         "can't convert uint to processorid_t");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   617
  int bind_result =
489c9b5090e2 Initial load
duke
parents:
diff changeset
   618
    processor_bind(P_LWPID,                       // bind LWP.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   619
                   P_MYID,                        // bind current LWP.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   620
                   (processorid_t) processor_id,  // id.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   621
                   NULL);                         // don't return old binding.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   622
  return (bind_result == 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   623
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   624
489c9b5090e2 Initial load
duke
parents:
diff changeset
   625
bool os::getenv(const char* name, char* buffer, int len) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   626
  char* val = ::getenv( name );
489c9b5090e2 Initial load
duke
parents:
diff changeset
   627
  if ( val == NULL
489c9b5090e2 Initial load
duke
parents:
diff changeset
   628
  ||   strlen(val) + 1  >  len ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   629
    if (len > 0)  buffer[0] = 0; // return a null string
489c9b5090e2 Initial load
duke
parents:
diff changeset
   630
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   631
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   632
  strcpy( buffer, val );
489c9b5090e2 Initial load
duke
parents:
diff changeset
   633
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   634
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   635
489c9b5090e2 Initial load
duke
parents:
diff changeset
   636
489c9b5090e2 Initial load
duke
parents:
diff changeset
   637
// Return true if user is running as root.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   638
489c9b5090e2 Initial load
duke
parents:
diff changeset
   639
bool os::have_special_privileges() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   640
  static bool init = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   641
  static bool privileges = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   642
  if (!init) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   643
    privileges = (getuid() != geteuid()) || (getgid() != getegid());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   644
    init = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   645
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   646
  return privileges;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   647
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   648
489c9b5090e2 Initial load
duke
parents:
diff changeset
   649
489c9b5090e2 Initial load
duke
parents:
diff changeset
   650
void os::init_system_properties_values() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   651
  char arch[12];
489c9b5090e2 Initial load
duke
parents:
diff changeset
   652
  sysinfo(SI_ARCHITECTURE, arch, sizeof(arch));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   653
489c9b5090e2 Initial load
duke
parents:
diff changeset
   654
  // The next steps are taken in the product version:
489c9b5090e2 Initial load
duke
parents:
diff changeset
   655
  //
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   656
  // Obtain the JAVA_HOME value from the location of libjvm.so.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   657
  // This library should be located at:
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   658
  // <JAVA_HOME>/jre/lib/<arch>/{client|server}/libjvm.so.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   659
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   660
  // If "/jre/lib/" appears at the right place in the path, then we
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   661
  // assume libjvm.so is installed in a JDK and we use this path.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   662
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   663
  // Otherwise exit with message: "Could not create the Java virtual machine."
489c9b5090e2 Initial load
duke
parents:
diff changeset
   664
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   665
  // The following extra steps are taken in the debugging version:
489c9b5090e2 Initial load
duke
parents:
diff changeset
   666
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   667
  // If "/jre/lib/" does NOT appear at the right place in the path
489c9b5090e2 Initial load
duke
parents:
diff changeset
   668
  // instead of exit check for $JAVA_HOME environment variable.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   669
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   670
  // If it is defined and we are able to locate $JAVA_HOME/jre/lib/<arch>,
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   671
  // then we append a fake suffix "hotspot/libjvm.so" to this path so
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   672
  // it looks like libjvm.so is installed there
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
   673
  // <JAVA_HOME>/jre/lib/<arch>/hotspot/libjvm.so.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   674
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   675
  // Otherwise exit.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   676
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   677
  // Important note: if the location of libjvm.so changes this
489c9b5090e2 Initial load
duke
parents:
diff changeset
   678
  // code needs to be changed accordingly.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   679
489c9b5090e2 Initial load
duke
parents:
diff changeset
   680
  // The next few definitions allow the code to be verbatim:
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   681
#define malloc(n) (char*)NEW_C_HEAP_ARRAY(char, (n), mtInternal)
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
   682
#define free(p) FREE_C_HEAP_ARRAY(char, p, mtInternal)
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   683
#define getenv(n) ::getenv(n)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   684
489c9b5090e2 Initial load
duke
parents:
diff changeset
   685
#define EXTENSIONS_DIR  "/lib/ext"
489c9b5090e2 Initial load
duke
parents:
diff changeset
   686
#define ENDORSED_DIR    "/lib/endorsed"
489c9b5090e2 Initial load
duke
parents:
diff changeset
   687
#define COMMON_DIR      "/usr/jdk/packages"
489c9b5090e2 Initial load
duke
parents:
diff changeset
   688
489c9b5090e2 Initial load
duke
parents:
diff changeset
   689
  {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   690
    /* sysclasspath, java_home, dll_dir */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   691
    {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   692
        char *home_path;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   693
        char *dll_path;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   694
        char *pslash;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   695
        char buf[MAXPATHLEN];
489c9b5090e2 Initial load
duke
parents:
diff changeset
   696
        os::jvm_path(buf, sizeof(buf));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   697
489c9b5090e2 Initial load
duke
parents:
diff changeset
   698
        // Found the full path to libjvm.so.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   699
        // Now cut the path to <java_home>/jre if we can.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   700
        *(strrchr(buf, '/')) = '\0';  /* get rid of /libjvm.so */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   701
        pslash = strrchr(buf, '/');
489c9b5090e2 Initial load
duke
parents:
diff changeset
   702
        if (pslash != NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   703
            *pslash = '\0';           /* get rid of /{client|server|hotspot} */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   704
        dll_path = malloc(strlen(buf) + 1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   705
        if (dll_path == NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   706
            return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   707
        strcpy(dll_path, buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   708
        Arguments::set_dll_dir(dll_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   709
489c9b5090e2 Initial load
duke
parents:
diff changeset
   710
        if (pslash != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   711
            pslash = strrchr(buf, '/');
489c9b5090e2 Initial load
duke
parents:
diff changeset
   712
            if (pslash != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   713
                *pslash = '\0';       /* get rid of /<arch> */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   714
                pslash = strrchr(buf, '/');
489c9b5090e2 Initial load
duke
parents:
diff changeset
   715
                if (pslash != NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   716
                    *pslash = '\0';   /* get rid of /lib */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   717
            }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   718
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   719
489c9b5090e2 Initial load
duke
parents:
diff changeset
   720
        home_path = malloc(strlen(buf) + 1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   721
        if (home_path == NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   722
            return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   723
        strcpy(home_path, buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   724
        Arguments::set_java_home(home_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   725
489c9b5090e2 Initial load
duke
parents:
diff changeset
   726
        if (!set_boot_path('/', ':'))
489c9b5090e2 Initial load
duke
parents:
diff changeset
   727
            return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   728
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   729
489c9b5090e2 Initial load
duke
parents:
diff changeset
   730
    /*
489c9b5090e2 Initial load
duke
parents:
diff changeset
   731
     * Where to look for native libraries
489c9b5090e2 Initial load
duke
parents:
diff changeset
   732
     */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   733
    {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   734
      // Use dlinfo() to determine the correct java.library.path.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   735
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   736
      // If we're launched by the Java launcher, and the user
489c9b5090e2 Initial load
duke
parents:
diff changeset
   737
      // does not set java.library.path explicitly on the commandline,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   738
      // the Java launcher sets LD_LIBRARY_PATH for us and unsets
489c9b5090e2 Initial load
duke
parents:
diff changeset
   739
      // LD_LIBRARY_PATH_32 and LD_LIBRARY_PATH_64.  In this case
489c9b5090e2 Initial load
duke
parents:
diff changeset
   740
      // dlinfo returns LD_LIBRARY_PATH + crle settings (including
489c9b5090e2 Initial load
duke
parents:
diff changeset
   741
      // /usr/lib), which is exactly what we want.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   742
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   743
      // If the user does set java.library.path, it completely
489c9b5090e2 Initial load
duke
parents:
diff changeset
   744
      // overwrites this setting, and always has.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   745
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   746
      // If we're not launched by the Java launcher, we may
489c9b5090e2 Initial load
duke
parents:
diff changeset
   747
      // get here with any/all of the LD_LIBRARY_PATH[_32|64]
489c9b5090e2 Initial load
duke
parents:
diff changeset
   748
      // settings.  Again, dlinfo does exactly what we want.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   749
489c9b5090e2 Initial load
duke
parents:
diff changeset
   750
      Dl_serinfo     _info, *info = &_info;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   751
      Dl_serpath     *path;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   752
      char*          library_path;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   753
      char           *common_path;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   754
      int            i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   755
489c9b5090e2 Initial load
duke
parents:
diff changeset
   756
      // determine search path count and required buffer size
489c9b5090e2 Initial load
duke
parents:
diff changeset
   757
      if (dlinfo(RTLD_SELF, RTLD_DI_SERINFOSIZE, (void *)info) == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   758
        vm_exit_during_initialization("dlinfo SERINFOSIZE request", dlerror());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   759
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   760
489c9b5090e2 Initial load
duke
parents:
diff changeset
   761
      // allocate new buffer and initialize
489c9b5090e2 Initial load
duke
parents:
diff changeset
   762
      info = (Dl_serinfo*)malloc(_info.dls_size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   763
      if (info == NULL) {
17087
f0b76c4c93a0 8011661: Insufficient memory message says "malloc" when sometimes it should say "mmap"
ccheung
parents: 16672
diff changeset
   764
        vm_exit_out_of_memory(_info.dls_size, OOM_MALLOC_ERROR,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   765
                              "init_system_properties_values info");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   766
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   767
      info->dls_size = _info.dls_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   768
      info->dls_cnt = _info.dls_cnt;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   769
489c9b5090e2 Initial load
duke
parents:
diff changeset
   770
      // obtain search path information
489c9b5090e2 Initial load
duke
parents:
diff changeset
   771
      if (dlinfo(RTLD_SELF, RTLD_DI_SERINFO, (void *)info) == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   772
        free(info);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   773
        vm_exit_during_initialization("dlinfo SERINFO request", dlerror());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   774
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   775
489c9b5090e2 Initial load
duke
parents:
diff changeset
   776
      path = &info->dls_serpath[0];
489c9b5090e2 Initial load
duke
parents:
diff changeset
   777
489c9b5090e2 Initial load
duke
parents:
diff changeset
   778
      // Note: Due to a legacy implementation, most of the library path
489c9b5090e2 Initial load
duke
parents:
diff changeset
   779
      // is set in the launcher.  This was to accomodate linking restrictions
489c9b5090e2 Initial load
duke
parents:
diff changeset
   780
      // on legacy Solaris implementations (which are no longer supported).
489c9b5090e2 Initial load
duke
parents:
diff changeset
   781
      // Eventually, all the library path setting will be done here.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   782
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   783
      // However, to prevent the proliferation of improperly built native
489c9b5090e2 Initial load
duke
parents:
diff changeset
   784
      // libraries, the new path component /usr/jdk/packages is added here.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   785
489c9b5090e2 Initial load
duke
parents:
diff changeset
   786
      // Determine the actual CPU architecture.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   787
      char cpu_arch[12];
489c9b5090e2 Initial load
duke
parents:
diff changeset
   788
      sysinfo(SI_ARCHITECTURE, cpu_arch, sizeof(cpu_arch));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   789
#ifdef _LP64
489c9b5090e2 Initial load
duke
parents:
diff changeset
   790
      // If we are a 64-bit vm, perform the following translations:
489c9b5090e2 Initial load
duke
parents:
diff changeset
   791
      //   sparc   -> sparcv9
489c9b5090e2 Initial load
duke
parents:
diff changeset
   792
      //   i386    -> amd64
489c9b5090e2 Initial load
duke
parents:
diff changeset
   793
      if (strcmp(cpu_arch, "sparc") == 0)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   794
        strcat(cpu_arch, "v9");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   795
      else if (strcmp(cpu_arch, "i386") == 0)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   796
        strcpy(cpu_arch, "amd64");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   797
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
   798
489c9b5090e2 Initial load
duke
parents:
diff changeset
   799
      // Construct the invariant part of ld_library_path. Note that the
489c9b5090e2 Initial load
duke
parents:
diff changeset
   800
      // space for the colon and the trailing null are provided by the
489c9b5090e2 Initial load
duke
parents:
diff changeset
   801
      // nulls included by the sizeof operator.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   802
      size_t bufsize = sizeof(COMMON_DIR) + sizeof("/lib/") + strlen(cpu_arch);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   803
      common_path = malloc(bufsize);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   804
      if (common_path == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   805
        free(info);
17087
f0b76c4c93a0 8011661: Insufficient memory message says "malloc" when sometimes it should say "mmap"
ccheung
parents: 16672
diff changeset
   806
        vm_exit_out_of_memory(bufsize, OOM_MALLOC_ERROR,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   807
                              "init_system_properties_values common_path");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   808
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   809
      sprintf(common_path, COMMON_DIR "/lib/%s", cpu_arch);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   810
489c9b5090e2 Initial load
duke
parents:
diff changeset
   811
      // struct size is more than sufficient for the path components obtained
489c9b5090e2 Initial load
duke
parents:
diff changeset
   812
      // through the dlinfo() call, so only add additional space for the path
489c9b5090e2 Initial load
duke
parents:
diff changeset
   813
      // components explicitly added here.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   814
      bufsize = info->dls_size + strlen(common_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   815
      library_path = malloc(bufsize);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   816
      if (library_path == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   817
        free(info);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   818
        free(common_path);
17087
f0b76c4c93a0 8011661: Insufficient memory message says "malloc" when sometimes it should say "mmap"
ccheung
parents: 16672
diff changeset
   819
        vm_exit_out_of_memory(bufsize, OOM_MALLOC_ERROR,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   820
                              "init_system_properties_values library_path");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   821
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   822
      library_path[0] = '\0';
489c9b5090e2 Initial load
duke
parents:
diff changeset
   823
489c9b5090e2 Initial load
duke
parents:
diff changeset
   824
      // Construct the desired Java library path from the linker's library
489c9b5090e2 Initial load
duke
parents:
diff changeset
   825
      // search path.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   826
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
   827
      // For compatibility, it is optimal that we insert the additional path
489c9b5090e2 Initial load
duke
parents:
diff changeset
   828
      // components specific to the Java VM after those components specified
489c9b5090e2 Initial load
duke
parents:
diff changeset
   829
      // in LD_LIBRARY_PATH (if any) but before those added by the ld.so
489c9b5090e2 Initial load
duke
parents:
diff changeset
   830
      // infrastructure.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   831
      if (info->dls_cnt == 0) { // Not sure this can happen, but allow for it
489c9b5090e2 Initial load
duke
parents:
diff changeset
   832
        strcpy(library_path, common_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   833
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   834
        int inserted = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   835
        for (i = 0; i < info->dls_cnt; i++, path++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   836
          uint_t flags = path->dls_flags & LA_SER_MASK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   837
          if (((flags & LA_SER_LIBPATH) == 0) && !inserted) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   838
            strcat(library_path, common_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   839
            strcat(library_path, os::path_separator());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   840
            inserted = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   841
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   842
          strcat(library_path, path->dls_name);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   843
          strcat(library_path, os::path_separator());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   844
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   845
        // eliminate trailing path separator
489c9b5090e2 Initial load
duke
parents:
diff changeset
   846
        library_path[strlen(library_path)-1] = '\0';
489c9b5090e2 Initial load
duke
parents:
diff changeset
   847
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   848
489c9b5090e2 Initial load
duke
parents:
diff changeset
   849
      // happens before argument parsing - can't use a trace flag
489c9b5090e2 Initial load
duke
parents:
diff changeset
   850
      // tty->print_raw("init_system_properties_values: native lib path: ");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   851
      // tty->print_raw_cr(library_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   852
489c9b5090e2 Initial load
duke
parents:
diff changeset
   853
      // callee copies into its own buffer
489c9b5090e2 Initial load
duke
parents:
diff changeset
   854
      Arguments::set_library_path(library_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   855
489c9b5090e2 Initial load
duke
parents:
diff changeset
   856
      free(common_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   857
      free(library_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   858
      free(info);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   859
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   860
489c9b5090e2 Initial load
duke
parents:
diff changeset
   861
    /*
489c9b5090e2 Initial load
duke
parents:
diff changeset
   862
     * Extensions directories.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   863
     *
489c9b5090e2 Initial load
duke
parents:
diff changeset
   864
     * Note that the space for the colon and the trailing null are provided
489c9b5090e2 Initial load
duke
parents:
diff changeset
   865
     * by the nulls included by the sizeof operator (so actually one byte more
489c9b5090e2 Initial load
duke
parents:
diff changeset
   866
     * than necessary is allocated).
489c9b5090e2 Initial load
duke
parents:
diff changeset
   867
     */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   868
    {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   869
        char *buf = (char *) malloc(strlen(Arguments::get_java_home()) +
489c9b5090e2 Initial load
duke
parents:
diff changeset
   870
            sizeof(EXTENSIONS_DIR) + sizeof(COMMON_DIR) +
489c9b5090e2 Initial load
duke
parents:
diff changeset
   871
            sizeof(EXTENSIONS_DIR));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   872
        sprintf(buf, "%s" EXTENSIONS_DIR ":" COMMON_DIR EXTENSIONS_DIR,
489c9b5090e2 Initial load
duke
parents:
diff changeset
   873
            Arguments::get_java_home());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   874
        Arguments::set_ext_dirs(buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   875
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   876
489c9b5090e2 Initial load
duke
parents:
diff changeset
   877
    /* Endorsed standards default directory. */
489c9b5090e2 Initial load
duke
parents:
diff changeset
   878
    {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   879
        char * buf = malloc(strlen(Arguments::get_java_home()) + sizeof(ENDORSED_DIR));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   880
        sprintf(buf, "%s" ENDORSED_DIR, Arguments::get_java_home());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   881
        Arguments::set_endorsed_dirs(buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   882
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   883
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   884
489c9b5090e2 Initial load
duke
parents:
diff changeset
   885
#undef malloc
489c9b5090e2 Initial load
duke
parents:
diff changeset
   886
#undef free
489c9b5090e2 Initial load
duke
parents:
diff changeset
   887
#undef getenv
489c9b5090e2 Initial load
duke
parents:
diff changeset
   888
#undef EXTENSIONS_DIR
489c9b5090e2 Initial load
duke
parents:
diff changeset
   889
#undef ENDORSED_DIR
489c9b5090e2 Initial load
duke
parents:
diff changeset
   890
#undef COMMON_DIR
489c9b5090e2 Initial load
duke
parents:
diff changeset
   891
489c9b5090e2 Initial load
duke
parents:
diff changeset
   892
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   893
489c9b5090e2 Initial load
duke
parents:
diff changeset
   894
void os::breakpoint() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   895
  BREAKPOINT;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   896
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   897
489c9b5090e2 Initial load
duke
parents:
diff changeset
   898
bool os::obsolete_option(const JavaVMOption *option)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   899
{
489c9b5090e2 Initial load
duke
parents:
diff changeset
   900
  if (!strncmp(option->optionString, "-Xt", 3)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   901
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   902
  } else if (!strncmp(option->optionString, "-Xtm", 4)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   903
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   904
  } else if (!strncmp(option->optionString, "-Xverifyheap", 12)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   905
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   906
  } else if (!strncmp(option->optionString, "-Xmaxjitcodesize", 16)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   907
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   908
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   909
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   910
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   911
489c9b5090e2 Initial load
duke
parents:
diff changeset
   912
bool os::Solaris::valid_stack_address(Thread* thread, address sp) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   913
  address  stackStart  = (address)thread->stack_base();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   914
  address  stackEnd    = (address)(stackStart - (address)thread->stack_size());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   915
  if (sp < stackStart && sp >= stackEnd ) return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   916
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   917
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   918
489c9b5090e2 Initial load
duke
parents:
diff changeset
   919
extern "C" void breakpoint() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   920
  // use debugger to set breakpoint here
489c9b5090e2 Initial load
duke
parents:
diff changeset
   921
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   922
489c9b5090e2 Initial load
duke
parents:
diff changeset
   923
static thread_t main_thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   924
489c9b5090e2 Initial load
duke
parents:
diff changeset
   925
// Thread start routine for all new Java threads
489c9b5090e2 Initial load
duke
parents:
diff changeset
   926
extern "C" void* java_start(void* thread_addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   927
  // Try to randomize the cache line index of hot stack frames.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   928
  // This helps when threads of the same stack traces evict each other's
489c9b5090e2 Initial load
duke
parents:
diff changeset
   929
  // cache lines. The threads can be either from the same JVM instance, or
489c9b5090e2 Initial load
duke
parents:
diff changeset
   930
  // from different JVM instances. The benefit is especially true for
489c9b5090e2 Initial load
duke
parents:
diff changeset
   931
  // processors with hyperthreading technology.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   932
  static int counter = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   933
  int pid = os::current_process_id();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   934
  alloca(((pid ^ counter++) & 7) * 128);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   935
489c9b5090e2 Initial load
duke
parents:
diff changeset
   936
  int prio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   937
  Thread* thread = (Thread*)thread_addr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   938
  OSThread* osthr = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   939
489c9b5090e2 Initial load
duke
parents:
diff changeset
   940
  osthr->set_lwp_id( _lwp_self() );  // Store lwp in case we are bound
489c9b5090e2 Initial load
duke
parents:
diff changeset
   941
  thread->_schedctl = (void *) schedctl_init () ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   942
489c9b5090e2 Initial load
duke
parents:
diff changeset
   943
  if (UseNUMA) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   944
    int lgrp_id = os::numa_get_group_id();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   945
    if (lgrp_id != -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   946
      thread->set_lgrp_id(lgrp_id);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   947
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   948
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   949
489c9b5090e2 Initial load
duke
parents:
diff changeset
   950
  // If the creator called set priority before we started,
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   951
  // we need to call set_native_priority now that we have an lwp.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   952
  // We used to get the priority from thr_getprio (we called
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   953
  // thr_setprio way back in create_thread) and pass it to
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   954
  // set_native_priority, but Solaris scales the priority
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   955
  // in java_to_os_priority, so when we read it back here,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   956
  // we pass trash to set_native_priority instead of what's
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   957
  // in java_to_os_priority. So we save the native priority
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   958
  // in the osThread and recall it here.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   959
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   960
  if ( osthr->thread_id() != -1 ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   961
    if ( UseThreadPriorities ) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   962
      int prio = osthr->native_priority();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   963
      if (ThreadPriorityVerbose) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   964
        tty->print_cr("Starting Thread " INTPTR_FORMAT ", LWP is "
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   965
                      INTPTR_FORMAT ", setting priority: %d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
   966
                      osthr->thread_id(), osthr->lwp_id(), prio);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   967
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   968
      os::set_native_priority(thread, prio);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   969
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   970
  } else if (ThreadPriorityVerbose) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   971
    warning("Can't set priority in _start routine, thread id hasn't been set\n");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   972
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   973
489c9b5090e2 Initial load
duke
parents:
diff changeset
   974
  assert(osthr->get_state() == RUNNABLE, "invalid os thread state");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   975
489c9b5090e2 Initial load
duke
parents:
diff changeset
   976
  // initialize signal mask for this thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
   977
  os::Solaris::hotspot_sigmask(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   978
489c9b5090e2 Initial load
duke
parents:
diff changeset
   979
  thread->run();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   980
489c9b5090e2 Initial load
duke
parents:
diff changeset
   981
  // One less thread is executing
489c9b5090e2 Initial load
duke
parents:
diff changeset
   982
  // When the VMThread gets here, the main thread may have already exited
489c9b5090e2 Initial load
duke
parents:
diff changeset
   983
  // which frees the CodeHeap containing the Atomic::dec code
489c9b5090e2 Initial load
duke
parents:
diff changeset
   984
  if (thread != VMThread::vm_thread() && VMThread::vm_thread() != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   985
    Atomic::dec(&os::Solaris::_os_thread_count);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   986
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   987
489c9b5090e2 Initial load
duke
parents:
diff changeset
   988
  if (UseDetachedThreads) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   989
    thr_exit(NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   990
    ShouldNotReachHere();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   991
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   992
  return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   993
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
   994
489c9b5090e2 Initial load
duke
parents:
diff changeset
   995
static OSThread* create_os_thread(Thread* thread, thread_t thread_id) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   996
  // Allocate the OSThread object
489c9b5090e2 Initial load
duke
parents:
diff changeset
   997
  OSThread* osthread = new OSThread(NULL, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   998
  if (osthread == NULL) return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
   999
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1000
  // Store info on the Solaris thread into the OSThread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1001
  osthread->set_thread_id(thread_id);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1002
  osthread->set_lwp_id(_lwp_self());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1003
  thread->_schedctl = (void *) schedctl_init () ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1004
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1005
  if (UseNUMA) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1006
    int lgrp_id = os::numa_get_group_id();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1007
    if (lgrp_id != -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1008
      thread->set_lgrp_id(lgrp_id);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1009
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1010
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1011
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1012
  if ( ThreadPriorityVerbose ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1013
    tty->print_cr("In create_os_thread, Thread " INTPTR_FORMAT ", LWP is " INTPTR_FORMAT "\n",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1014
                  osthread->thread_id(), osthread->lwp_id() );
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1015
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1016
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1017
  // Initial thread state is INITIALIZED, not SUSPENDED
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1018
  osthread->set_state(INITIALIZED);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1019
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1020
  return osthread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1021
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1022
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1023
void os::Solaris::hotspot_sigmask(Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1024
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1025
  //Save caller's signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1026
  sigset_t sigmask;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1027
  thr_sigsetmask(SIG_SETMASK, NULL, &sigmask);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1028
  OSThread *osthread = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1029
  osthread->set_caller_sigmask(sigmask);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1030
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1031
  thr_sigsetmask(SIG_UNBLOCK, os::Solaris::unblocked_signals(), NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1032
  if (!ReduceSignalUsage) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1033
    if (thread->is_VM_thread()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1034
      // Only the VM thread handles BREAK_SIGNAL ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1035
      thr_sigsetmask(SIG_UNBLOCK, vm_signals(), NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1036
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1037
      // ... all other threads block BREAK_SIGNAL
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1038
      assert(!sigismember(vm_signals(), SIGINT), "SIGINT should not be blocked");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1039
      thr_sigsetmask(SIG_BLOCK, vm_signals(), NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1040
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1041
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1042
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1043
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1044
bool os::create_attached_thread(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1045
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1046
  thread->verify_not_published();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1047
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1048
  OSThread* osthread = create_os_thread(thread, thr_self());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1049
  if (osthread == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1050
     return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1051
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1052
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1053
  // Initial thread state is RUNNABLE
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1054
  osthread->set_state(RUNNABLE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1055
  thread->set_osthread(osthread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1056
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1057
  // initialize signal mask for this thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1058
  // and save the caller's signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1059
  os::Solaris::hotspot_sigmask(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1060
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1061
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1062
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1063
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1064
bool os::create_main_thread(JavaThread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1065
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1066
  thread->verify_not_published();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1067
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1068
  if (_starting_thread == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1069
    _starting_thread = create_os_thread(thread, main_thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1070
     if (_starting_thread == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1071
        return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1072
     }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1073
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1074
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1075
  // The primodial thread is runnable from the start
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1076
  _starting_thread->set_state(RUNNABLE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1077
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1078
  thread->set_osthread(_starting_thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1079
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1080
  // initialize signal mask for this thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1081
  // and save the caller's signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1082
  os::Solaris::hotspot_sigmask(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1083
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1084
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1085
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1086
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1087
// _T2_libthread is true if we believe we are running with the newer
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1088
// SunSoft lwp/libthread.so (2.8 patch, 2.9 default)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1089
bool os::Solaris::_T2_libthread = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1090
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1091
bool os::create_thread(Thread* thread, ThreadType thr_type, size_t stack_size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1092
  // Allocate the OSThread object
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1093
  OSThread* osthread = new OSThread(NULL, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1094
  if (osthread == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1095
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1096
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1097
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1098
  if ( ThreadPriorityVerbose ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1099
    char *thrtyp;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1100
    switch ( thr_type ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1101
      case vm_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1102
        thrtyp = (char *)"vm";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1103
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1104
      case cgc_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1105
        thrtyp = (char *)"cgc";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1106
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1107
      case pgc_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1108
        thrtyp = (char *)"pgc";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1109
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1110
      case java_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1111
        thrtyp = (char *)"java";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1112
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1113
      case compiler_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1114
        thrtyp = (char *)"compiler";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1115
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1116
      case watcher_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1117
        thrtyp = (char *)"watcher";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1118
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1119
      default:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1120
        thrtyp = (char *)"unknown";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1121
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1122
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1123
    tty->print_cr("In create_thread, creating a %s thread\n", thrtyp);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1124
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1125
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1126
  // Calculate stack size if it's not specified by caller.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1127
  if (stack_size == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1128
    // The default stack size 1M (2M for LP64).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1129
    stack_size = (BytesPerWord >> 2) * K * K;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1130
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1131
    switch (thr_type) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1132
    case os::java_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1133
      // Java threads use ThreadStackSize which default value can be changed with the flag -Xss
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1134
      if (JavaThread::stack_size_at_create() > 0) stack_size = JavaThread::stack_size_at_create();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1135
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1136
    case os::compiler_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1137
      if (CompilerThreadStackSize > 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1138
        stack_size = (size_t)(CompilerThreadStackSize * K);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1139
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1140
      } // else fall through:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1141
        // use VMThreadStackSize if CompilerThreadStackSize is not defined
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1142
    case os::vm_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1143
    case os::pgc_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1144
    case os::cgc_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1145
    case os::watcher_thread:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1146
      if (VMThreadStackSize > 0) stack_size = (size_t)(VMThreadStackSize * K);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1147
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1148
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1149
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1150
  stack_size = MAX2(stack_size, os::Solaris::min_stack_allowed);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1151
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1152
  // Initial state is ALLOCATED but not INITIALIZED
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1153
  osthread->set_state(ALLOCATED);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1154
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1155
  if (os::Solaris::_os_thread_count > os::Solaris::_os_thread_limit) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1156
    // We got lots of threads. Check if we still have some address space left.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1157
    // Need to be at least 5Mb of unreserved address space. We do check by
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1158
    // trying to reserve some.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1159
    const size_t VirtualMemoryBangSize = 20*K*K;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1160
    char* mem = os::reserve_memory(VirtualMemoryBangSize);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1161
    if (mem == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1162
      delete osthread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1163
      return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1164
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1165
      // Release the memory again
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1166
      os::release_memory(mem, VirtualMemoryBangSize);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1167
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1168
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1169
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1170
  // Setup osthread because the child thread may need it.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1171
  thread->set_osthread(osthread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1172
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1173
  // Create the Solaris thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1174
  // explicit THR_BOUND for T2_libthread case in case
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1175
  // that assumption is not accurate, but our alternate signal stack
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1176
  // handling is based on it which must have bound threads
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1177
  thread_t tid = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1178
  long     flags = (UseDetachedThreads ? THR_DETACHED : 0) | THR_SUSPENDED
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1179
                   | ((UseBoundThreads || os::Solaris::T2_libthread() ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1180
                       (thr_type == vm_thread) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1181
                       (thr_type == cgc_thread) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1182
                       (thr_type == pgc_thread) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1183
                       (thr_type == compiler_thread && BackgroundCompilation)) ?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1184
                      THR_BOUND : 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1185
  int      status;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1186
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1187
  // 4376845 -- libthread/kernel don't provide enough LWPs to utilize all CPUs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1188
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1189
  // On multiprocessors systems, libthread sometimes under-provisions our
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1190
  // process with LWPs.  On a 30-way systems, for instance, we could have
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1191
  // 50 user-level threads in ready state and only 2 or 3 LWPs assigned
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1192
  // to our process.  This can result in under utilization of PEs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1193
  // I suspect the problem is related to libthread's LWP
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1194
  // pool management and to the kernel's SIGBLOCKING "last LWP parked"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1195
  // upcall policy.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1196
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1197
  // The following code is palliative -- it attempts to ensure that our
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1198
  // process has sufficient LWPs to take advantage of multiple PEs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1199
  // Proper long-term cures include using user-level threads bound to LWPs
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1200
  // (THR_BOUND) or using LWP-based synchronization.  Note that there is a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1201
  // slight timing window with respect to sampling _os_thread_count, but
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1202
  // the race is benign.  Also, we should periodically recompute
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1203
  // _processors_online as the min of SC_NPROCESSORS_ONLN and the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1204
  // the number of PEs in our partition.  You might be tempted to use
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1205
  // THR_NEW_LWP here, but I'd recommend against it as that could
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1206
  // result in undesirable growth of the libthread's LWP pool.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1207
  // The fix below isn't sufficient; for instance, it doesn't take into count
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1208
  // LWPs parked on IO.  It does, however, help certain CPU-bound benchmarks.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1209
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1210
  // Some pathologies this scheme doesn't handle:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1211
  // *  Threads can block, releasing the LWPs.  The LWPs can age out.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1212
  //    When a large number of threads become ready again there aren't
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1213
  //    enough LWPs available to service them.  This can occur when the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1214
  //    number of ready threads oscillates.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1215
  // *  LWPs/Threads park on IO, thus taking the LWP out of circulation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1216
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1217
  // Finally, we should call thr_setconcurrency() periodically to refresh
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1218
  // the LWP pool and thwart the LWP age-out mechanism.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1219
  // The "+3" term provides a little slop -- we want to slightly overprovision.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1220
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1221
  if (AdjustConcurrency && os::Solaris::_os_thread_count < (_processors_online+3)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1222
    if (!(flags & THR_BOUND)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1223
      thr_setconcurrency (os::Solaris::_os_thread_count);       // avoid starvation
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1224
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1225
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1226
  // Although this doesn't hurt, we should warn of undefined behavior
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1227
  // when using unbound T1 threads with schedctl().  This should never
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1228
  // happen, as the compiler and VM threads are always created bound
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1229
  DEBUG_ONLY(
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1230
      if ((VMThreadHintNoPreempt || CompilerThreadHintNoPreempt) &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1231
          (!os::Solaris::T2_libthread() && (!(flags & THR_BOUND))) &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1232
          ((thr_type == vm_thread) || (thr_type == cgc_thread) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1233
           (thr_type == pgc_thread) || (thr_type == compiler_thread && BackgroundCompilation))) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1234
         warning("schedctl behavior undefined when Compiler/VM/GC Threads are Unbound");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1235
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1236
  );
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1237
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1238
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1239
  // Mark that we don't have an lwp or thread id yet.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1240
  // In case we attempt to set the priority before the thread starts.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1241
  osthread->set_lwp_id(-1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1242
  osthread->set_thread_id(-1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1243
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1244
  status = thr_create(NULL, stack_size, java_start, thread, flags, &tid);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1245
  if (status != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1246
    if (PrintMiscellaneous && (Verbose || WizardMode)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1247
      perror("os::create_thread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1248
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1249
    thread->set_osthread(NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1250
    // Need to clean up stuff we've allocated so far
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1251
    delete osthread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1252
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1253
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1254
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1255
  Atomic::inc(&os::Solaris::_os_thread_count);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1256
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1257
  // Store info on the Solaris thread into the OSThread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1258
  osthread->set_thread_id(tid);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1259
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1260
  // Remember that we created this thread so we can set priority on it
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1261
  osthread->set_vm_created();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1262
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  1263
  // Set the default thread priority.  If using bound threads, setting
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  1264
  // lwp priority will be delayed until thread start.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  1265
  set_native_priority(thread,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  1266
                      DefaultThreadPriority == -1 ?
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1267
                        java_to_os_priority[NormPriority] :
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1268
                        DefaultThreadPriority);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1269
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1270
  // Initial thread state is INITIALIZED, not SUSPENDED
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1271
  osthread->set_state(INITIALIZED);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1272
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1273
  // The thread is returned suspended (in state INITIALIZED), and is started higher up in the call chain
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1274
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1275
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1276
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1277
/* defined for >= Solaris 10. This allows builds on earlier versions
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1278
 *  of Solaris to take advantage of the newly reserved Solaris JVM signals
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1279
 *  With SIGJVM1, SIGJVM2, INTERRUPT_SIGNAL is SIGJVM1, ASYNC_SIGNAL is SIGJVM2
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1280
 *  and -XX:+UseAltSigs does nothing since these should have no conflict
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1281
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1282
#if !defined(SIGJVM1)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1283
#define SIGJVM1 39
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1284
#define SIGJVM2 40
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1285
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1286
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1287
debug_only(static bool signal_sets_initialized = false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1288
static sigset_t unblocked_sigs, vm_sigs, allowdebug_blocked_sigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1289
int os::Solaris::_SIGinterrupt = INTERRUPT_SIGNAL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1290
int os::Solaris::_SIGasync = ASYNC_SIGNAL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1291
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1292
bool os::Solaris::is_sig_ignored(int sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1293
      struct sigaction oact;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1294
      sigaction(sig, (struct sigaction*)NULL, &oact);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1295
      void* ohlr = oact.sa_sigaction ? CAST_FROM_FN_PTR(void*,  oact.sa_sigaction)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1296
                                     : CAST_FROM_FN_PTR(void*,  oact.sa_handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1297
      if (ohlr == CAST_FROM_FN_PTR(void*, SIG_IGN))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1298
           return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1299
      else
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1300
           return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1301
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1302
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1303
// Note: SIGRTMIN is a macro that calls sysconf() so it will
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1304
// dynamically detect SIGRTMIN value for the system at runtime, not buildtime
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1305
static bool isJVM1available() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1306
  return SIGJVM1 < SIGRTMIN;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1307
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1308
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1309
void os::Solaris::signal_sets_init() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1310
  // Should also have an assertion stating we are still single-threaded.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1311
  assert(!signal_sets_initialized, "Already initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1312
  // Fill in signals that are necessarily unblocked for all threads in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1313
  // the VM. Currently, we unblock the following signals:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1314
  // SHUTDOWN{1,2,3}_SIGNAL: for shutdown hooks support (unless over-ridden
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1315
  //                         by -Xrs (=ReduceSignalUsage));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1316
  // BREAK_SIGNAL which is unblocked only by the VM thread and blocked by all
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1317
  // other threads. The "ReduceSignalUsage" boolean tells us not to alter
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1318
  // the dispositions or masks wrt these signals.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1319
  // Programs embedding the VM that want to use the above signals for their
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1320
  // own purposes must, at this time, use the "-Xrs" option to prevent
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1321
  // interference with shutdown hooks and BREAK_SIGNAL thread dumping.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1322
  // (See bug 4345157, and other related bugs).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1323
  // In reality, though, unblocking these signals is really a nop, since
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1324
  // these signals are not blocked by default.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1325
  sigemptyset(&unblocked_sigs);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1326
  sigemptyset(&allowdebug_blocked_sigs);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1327
  sigaddset(&unblocked_sigs, SIGILL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1328
  sigaddset(&unblocked_sigs, SIGSEGV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1329
  sigaddset(&unblocked_sigs, SIGBUS);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1330
  sigaddset(&unblocked_sigs, SIGFPE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1331
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1332
  if (isJVM1available) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1333
    os::Solaris::set_SIGinterrupt(SIGJVM1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1334
    os::Solaris::set_SIGasync(SIGJVM2);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1335
  } else if (UseAltSigs) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1336
    os::Solaris::set_SIGinterrupt(ALT_INTERRUPT_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1337
    os::Solaris::set_SIGasync(ALT_ASYNC_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1338
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1339
    os::Solaris::set_SIGinterrupt(INTERRUPT_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1340
    os::Solaris::set_SIGasync(ASYNC_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1341
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1342
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1343
  sigaddset(&unblocked_sigs, os::Solaris::SIGinterrupt());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1344
  sigaddset(&unblocked_sigs, os::Solaris::SIGasync());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1345
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1346
  if (!ReduceSignalUsage) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1347
   if (!os::Solaris::is_sig_ignored(SHUTDOWN1_SIGNAL)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1348
      sigaddset(&unblocked_sigs, SHUTDOWN1_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1349
      sigaddset(&allowdebug_blocked_sigs, SHUTDOWN1_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1350
   }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1351
   if (!os::Solaris::is_sig_ignored(SHUTDOWN2_SIGNAL)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1352
      sigaddset(&unblocked_sigs, SHUTDOWN2_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1353
      sigaddset(&allowdebug_blocked_sigs, SHUTDOWN2_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1354
   }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1355
   if (!os::Solaris::is_sig_ignored(SHUTDOWN3_SIGNAL)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1356
      sigaddset(&unblocked_sigs, SHUTDOWN3_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1357
      sigaddset(&allowdebug_blocked_sigs, SHUTDOWN3_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1358
   }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1359
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1360
  // Fill in signals that are blocked by all but the VM thread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1361
  sigemptyset(&vm_sigs);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1362
  if (!ReduceSignalUsage)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1363
    sigaddset(&vm_sigs, BREAK_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1364
  debug_only(signal_sets_initialized = true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1365
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1366
  // For diagnostics only used in run_periodic_checks
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1367
  sigemptyset(&check_signal_done);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1368
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1369
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1370
// These are signals that are unblocked while a thread is running Java.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1371
// (For some reason, they get blocked by default.)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1372
sigset_t* os::Solaris::unblocked_signals() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1373
  assert(signal_sets_initialized, "Not initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1374
  return &unblocked_sigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1375
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1376
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1377
// These are the signals that are blocked while a (non-VM) thread is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1378
// running Java. Only the VM thread handles these signals.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1379
sigset_t* os::Solaris::vm_signals() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1380
  assert(signal_sets_initialized, "Not initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1381
  return &vm_sigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1382
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1383
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1384
// These are signals that are blocked during cond_wait to allow debugger in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1385
sigset_t* os::Solaris::allowdebug_blocked_signals() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1386
  assert(signal_sets_initialized, "Not initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1387
  return &allowdebug_blocked_sigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1388
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1389
7692
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1390
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1391
void _handle_uncaught_cxx_exception() {
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1392
  VMError err("An uncaught C++ exception");
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1393
  err.report_and_die();
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1394
}
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1395
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1396
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1397
// First crack at OS-specific initialization, from inside the new thread.
13859
7fe4578493fc 7190089: NMT ON: NMT failed assertion on thread's stack base address
zgu
parents: 13728
diff changeset
  1398
void os::initialize_thread(Thread* thr) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1399
  int r = thr_main() ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1400
  guarantee (r == 0 || r == 1, "CR6501650 or CR6493689") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1401
  if (r) {
13859
7fe4578493fc 7190089: NMT ON: NMT failed assertion on thread's stack base address
zgu
parents: 13728
diff changeset
  1402
    JavaThread* jt = (JavaThread *)thr;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1403
    assert(jt != NULL,"Sanity check");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1404
    size_t stack_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1405
    address base = jt->stack_base();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1406
    if (Arguments::created_by_java_launcher()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1407
      // Use 2MB to allow for Solaris 7 64 bit mode.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1408
      stack_size = JavaThread::stack_size_at_create() == 0
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1409
        ? 2048*K : JavaThread::stack_size_at_create();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1410
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1411
      // There are rare cases when we may have already used more than
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1412
      // the basic stack size allotment before this method is invoked.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1413
      // Attempt to allow for a normally sized java_stack.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1414
      size_t current_stack_offset = (size_t)(base - (address)&stack_size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1415
      stack_size += ReservedSpace::page_align_size_down(current_stack_offset);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1416
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1417
      // 6269555: If we were not created by a Java launcher, i.e. if we are
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1418
      // running embedded in a native application, treat the primordial thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1419
      // as much like a native attached thread as possible.  This means using
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1420
      // the current stack size from thr_stksegment(), unless it is too large
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1421
      // to reliably setup guard pages.  A reasonable max size is 8MB.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1422
      size_t current_size = current_stack_size();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1423
      // This should never happen, but just in case....
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1424
      if (current_size == 0) current_size = 2 * K * K;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1425
      stack_size = current_size > (8 * K * K) ? (8 * K * K) : current_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1426
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1427
    address bottom = (address)align_size_up((intptr_t)(base - stack_size), os::vm_page_size());;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1428
    stack_size = (size_t)(base - bottom);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1429
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1430
    assert(stack_size > 0, "Stack size calculation problem");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1431
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1432
    if (stack_size > jt->stack_size()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1433
      NOT_PRODUCT(
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1434
        struct rlimit limits;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1435
        getrlimit(RLIMIT_STACK, &limits);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1436
        size_t size = adjust_stack_size(base, (size_t)limits.rlim_cur);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1437
        assert(size >= jt->stack_size(), "Stack size problem in main thread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1438
      )
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1439
      tty->print_cr(
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1440
        "Stack size of %d Kb exceeds current limit of %d Kb.\n"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1441
        "(Stack sizes are rounded up to a multiple of the system page size.)\n"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1442
        "See limit(1) to increase the stack size limit.",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1443
        stack_size / K, jt->stack_size() / K);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1444
      vm_exit(1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1445
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1446
    assert(jt->stack_size() >= stack_size,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1447
          "Attempt to map more stack than was allocated");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1448
    jt->set_stack_size(stack_size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1449
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1450
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1451
   // 5/22/01: Right now alternate signal stacks do not handle
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1452
   // throwing stack overflow exceptions, see bug 4463178
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1453
   // Until a fix is found for this, T2 will NOT imply alternate signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1454
   // stacks.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1455
   // If using T2 libthread threads, install an alternate signal stack.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1456
   // Because alternate stacks associate with LWPs on Solaris,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1457
   // see sigaltstack(2), if using UNBOUND threads, or if UseBoundThreads
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1458
   // we prefer to explicitly stack bang.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1459
   // If not using T2 libthread, but using UseBoundThreads any threads
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1460
   // (primordial thread, jni_attachCurrentThread) we do not create,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1461
   // probably are not bound, therefore they can not have an alternate
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1462
   // signal stack. Since our stack banging code is generated and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1463
   // is shared across threads, all threads must be bound to allow
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1464
   // using alternate signal stacks.  The alternative is to interpose
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1465
   // on _lwp_create to associate an alt sig stack with each LWP,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1466
   // and this could be a problem when the JVM is embedded.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1467
   // We would prefer to use alternate signal stacks with T2
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1468
   // Since there is currently no accurate way to detect T2
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1469
   // we do not. Assuming T2 when running T1 causes sig 11s or assertions
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1470
   // on installing alternate signal stacks
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1471
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1472
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1473
   // 05/09/03: removed alternate signal stack support for Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1474
   // The alternate signal stack mechanism is no longer needed to
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1475
   // handle stack overflow. This is now handled by allocating
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1476
   // guard pages (red zone) and stackbanging.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1477
   // Initially the alternate signal stack mechanism was removed because
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1478
   // it did not work with T1 llibthread. Alternate
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1479
   // signal stacks MUST have all threads bound to lwps. Applications
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1480
   // can create their own threads and attach them without their being
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1481
   // bound under T1. This is frequently the case for the primordial thread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1482
   // If we were ever to reenable this mechanism we would need to
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1483
   // use the dynamic check for T2 libthread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1484
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1485
  os::Solaris::init_thread_fpu_state();
7692
02c573fae027 6961186: Better VM handling of unexpected exceptions from application native code
zgu
parents: 7448
diff changeset
  1486
  std::set_terminate(_handle_uncaught_cxx_exception);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1487
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1488
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1489
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1490
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1491
// Free Solaris resources related to the OSThread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1492
void os::free_thread(OSThread* osthread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1493
  assert(osthread != NULL, "os::free_thread but osthread not set");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1494
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1495
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1496
  // We are told to free resources of the argument thread,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1497
  // but we can only really operate on the current thread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1498
  // The main thread must take the VMThread down synchronously
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1499
  // before the main thread exits and frees up CodeHeap
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1500
  guarantee((Thread::current()->osthread() == osthread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1501
     || (osthread == VMThread::vm_thread()->osthread())), "os::free_thread but not current thread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1502
  if (Thread::current()->osthread() == osthread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1503
    // Restore caller's signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1504
    sigset_t sigmask = osthread->caller_sigmask();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1505
    thr_sigsetmask(SIG_SETMASK, &sigmask, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1506
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1507
  delete osthread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1508
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1509
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1510
void os::pd_start_thread(Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1511
  int status = thr_continue(thread->osthread()->thread_id());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1512
  assert_status(status == 0, status, "thr_continue failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1513
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1514
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1515
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1516
intx os::current_thread_id() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1517
  return (intx)thr_self();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1518
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1519
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1520
static pid_t _initial_pid = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1521
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1522
int os::current_process_id() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1523
  return (int)(_initial_pid ? _initial_pid : getpid());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1524
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1525
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1526
int os::allocate_thread_local_storage() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1527
  // %%%       in Win32 this allocates a memory segment pointed to by a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1528
  //           register.  Dan Stein can implement a similar feature in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1529
  //           Solaris.  Alternatively, the VM can do the same thing
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1530
  //           explicitly: malloc some storage and keep the pointer in a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1531
  //           register (which is part of the thread's context) (or keep it
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1532
  //           in TLS).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1533
  // %%%       In current versions of Solaris, thr_self and TSD can
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1534
  //           be accessed via short sequences of displaced indirections.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1535
  //           The value of thr_self is available as %g7(36).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1536
  //           The value of thr_getspecific(k) is stored in %g7(12)(4)(k*4-4),
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1537
  //           assuming that the current thread already has a value bound to k.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1538
  //           It may be worth experimenting with such access patterns,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1539
  //           and later having the parameters formally exported from a Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1540
  //           interface.  I think, however, that it will be faster to
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1541
  //           maintain the invariant that %g2 always contains the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1542
  //           JavaThread in Java code, and have stubs simply
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1543
  //           treat %g2 as a caller-save register, preserving it in a %lN.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1544
  thread_key_t tk;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1545
  if (thr_keycreate( &tk, NULL ) )
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  1546
    fatal(err_msg("os::allocate_thread_local_storage: thr_keycreate failed "
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  1547
                  "(%s)", strerror(errno)));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1548
  return int(tk);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1549
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1550
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1551
void os::free_thread_local_storage(int index) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1552
  // %%% don't think we need anything here
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1553
  // if ( pthread_key_delete((pthread_key_t) tk) )
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1554
  //   fatal("os::free_thread_local_storage: pthread_key_delete failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1555
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1556
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1557
#define SMALLINT 32   // libthread allocate for tsd_common is a version specific
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1558
                      // small number - point is NO swap space available
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1559
void os::thread_local_storage_at_put(int index, void* value) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1560
  // %%% this is used only in threadLocalStorage.cpp
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1561
  if (thr_setspecific((thread_key_t)index, value)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1562
    if (errno == ENOMEM) {
17087
f0b76c4c93a0 8011661: Insufficient memory message says "malloc" when sometimes it should say "mmap"
ccheung
parents: 16672
diff changeset
  1563
       vm_exit_out_of_memory(SMALLINT, OOM_MALLOC_ERROR,
f0b76c4c93a0 8011661: Insufficient memory message says "malloc" when sometimes it should say "mmap"
ccheung
parents: 16672
diff changeset
  1564
                             "thr_setspecific: out of swap space");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1565
    } else {
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  1566
      fatal(err_msg("os::thread_local_storage_at_put: thr_setspecific failed "
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  1567
                    "(%s)", strerror(errno)));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1568
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1569
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1570
      ThreadLocalStorage::set_thread_in_slot ((Thread *) value) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1571
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1572
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1573
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1574
// This function could be called before TLS is initialized, for example, when
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1575
// VM receives an async signal or when VM causes a fatal error during
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1576
// initialization. Return NULL if thr_getspecific() fails.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1577
void* os::thread_local_storage_at(int index) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1578
  // %%% this is used only in threadLocalStorage.cpp
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1579
  void* r = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1580
  return thr_getspecific((thread_key_t)index, &r) != 0 ? NULL : r;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1581
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1582
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1583
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1584
// gethrtime can move backwards if read from one cpu and then a different cpu
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1585
// getTimeNanos is guaranteed to not move backward on Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1586
// local spinloop created as faster for a CAS on an int than
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1587
// a CAS on a 64bit jlong. Also Atomic::cmpxchg for jlong is not
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1588
// supported on sparc v8 or pre supports_cx8 intel boxes.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1589
// oldgetTimeNanos for systems which do not support CAS on 64bit jlong
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1590
// i.e. sparc v8 and pre supports_cx8 (i486) intel boxes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1591
inline hrtime_t oldgetTimeNanos() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1592
  int gotlock = LOCK_INVALID;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1593
  hrtime_t newtime = gethrtime();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1594
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1595
  for (;;) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1596
// grab lock for max_hrtime
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1597
    int curlock = max_hrtime_lock;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1598
    if (curlock & LOCK_BUSY)  continue;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1599
    if (gotlock = Atomic::cmpxchg(LOCK_BUSY, &max_hrtime_lock, LOCK_FREE) != LOCK_FREE) continue;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1600
    if (newtime > max_hrtime) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1601
      max_hrtime = newtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1602
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1603
      newtime = max_hrtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1604
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1605
    // release lock
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1606
    max_hrtime_lock = LOCK_FREE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1607
    return newtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1608
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1609
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1610
// gethrtime can move backwards if read from one cpu and then a different cpu
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1611
// getTimeNanos is guaranteed to not move backward on Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1612
inline hrtime_t getTimeNanos() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1613
  if (VM_Version::supports_cx8()) {
1686
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1614
    const hrtime_t now = gethrtime();
3594
0ce9158bc84c 6863420: os::javaTimeNanos() go backward on Solaris x86
kvn
parents: 2358
diff changeset
  1615
    // Use atomic long load since 32-bit x86 uses 2 registers to keep long.
0ce9158bc84c 6863420: os::javaTimeNanos() go backward on Solaris x86
kvn
parents: 2358
diff changeset
  1616
    const hrtime_t prev = Atomic::load((volatile jlong*)&max_hrtime);
1686
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1617
    if (now <= prev)  return prev;   // same or retrograde time;
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1618
    const hrtime_t obsv = Atomic::cmpxchg(now, (volatile jlong*)&max_hrtime, prev);
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1619
    assert(obsv >= prev, "invariant");   // Monotonicity
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1620
    // If the CAS succeeded then we're done and return "now".
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1621
    // If the CAS failed and the observed value "obs" is >= now then
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1622
    // we should return "obs".  If the CAS failed and now > obs > prv then
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1623
    // some other thread raced this thread and installed a new value, in which case
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1624
    // we could either (a) retry the entire operation, (b) retry trying to install now
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1625
    // or (c) just return obs.  We use (c).   No loop is required although in some cases
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1626
    // we might discard a higher "now" value in deference to a slightly lower but freshly
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1627
    // installed obs value.   That's entirely benign -- it admits no new orderings compared
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1628
    // to (a) or (b) -- and greatly reduces coherence traffic.
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1629
    // We might also condition (c) on the magnitude of the delta between obs and now.
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1630
    // Avoiding excessive CAS operations to hot RW locations is critical.
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1631
    // See http://blogs.sun.com/dave/entry/cas_and_cache_trivia_invalidate
27651ead9d6e 6784100: getTimeNanos - CAS reduction
xlu
parents: 1664
diff changeset
  1632
    return (prev == obsv) ? now : obsv ;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1633
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1634
    return oldgetTimeNanos();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1635
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1636
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1637
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1638
// Time since start-up in seconds to a fine granularity.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1639
// Used by VMSelfDestructTimer and the MemProfiler.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1640
double os::elapsedTime() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1641
  return (double)(getTimeNanos() - first_hrtime) / (double)hrtime_hz;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1642
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1643
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1644
jlong os::elapsed_counter() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1645
  return (jlong)(getTimeNanos() - first_hrtime);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1646
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1647
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1648
jlong os::elapsed_frequency() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1649
   return hrtime_hz;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1650
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1651
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1652
// Return the real, user, and system times in seconds from an
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1653
// arbitrary fixed point in the past.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1654
bool os::getTimesSecs(double* process_real_time,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1655
                  double* process_user_time,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1656
                  double* process_system_time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1657
  struct tms ticks;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1658
  clock_t real_ticks = times(&ticks);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1659
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1660
  if (real_ticks == (clock_t) (-1)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1661
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1662
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1663
    double ticks_per_second = (double) clock_tics_per_sec;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1664
    *process_user_time = ((double) ticks.tms_utime) / ticks_per_second;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1665
    *process_system_time = ((double) ticks.tms_stime) / ticks_per_second;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1666
    // For consistency return the real time from getTimeNanos()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1667
    // converted to seconds.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1668
    *process_real_time = ((double) getTimeNanos()) / ((double) NANOUNITS);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1669
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1670
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1671
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1672
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1673
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1674
bool os::supports_vtime() { return true; }
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1675
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1676
bool os::enable_vtime() {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1677
  int fd = ::open("/proc/self/ctl", O_WRONLY);
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1678
  if (fd == -1)
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1679
    return false;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1680
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1681
  long cmd[] = { PCSET, PR_MSACCT };
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1682
  int res = ::write(fd, cmd, sizeof(long) * 2);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1683
  ::close(fd);
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1684
  if (res != sizeof(long) * 2)
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1685
    return false;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1686
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1687
  return true;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1688
}
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1689
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1690
bool os::vtime_enabled() {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1691
  int fd = ::open("/proc/self/status", O_RDONLY);
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1692
  if (fd == -1)
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1693
    return false;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1694
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1695
  pstatus_t status;
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1696
  int res = os::read(fd, (void*) &status, sizeof(pstatus_t));
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  1697
  ::close(fd);
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1698
  if (res != sizeof(pstatus_t))
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1699
    return false;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1700
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1701
  return status.pr_flags & PR_MSACCT;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1702
}
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1703
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1704
double os::elapsedVTime() {
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1705
  return (double)gethrvtime() / (double)hrtime_hz;
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1706
}
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  1707
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1708
// Used internally for comparisons only
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1709
// getTimeMillis guaranteed to not move backwards on Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1710
jlong getTimeMillis() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1711
  jlong nanotime = getTimeNanos();
11251
e29da6b5622b 7117303: VM uses non-monotonic time source and complains that it is non-monotonic
johnc
parents: 11161
diff changeset
  1712
  return (jlong)(nanotime / NANOSECS_PER_MILLISEC);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1713
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1714
234
4da9c1bbc810 6667833: Remove CacheTimeMillis
sbohne
parents: 233
diff changeset
  1715
// Must return millis since Jan 1 1970 for JVM_CurrentTimeMillis
4da9c1bbc810 6667833: Remove CacheTimeMillis
sbohne
parents: 233
diff changeset
  1716
jlong os::javaTimeMillis() {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1717
  timeval t;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1718
  if (gettimeofday( &t, NULL) == -1)
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  1719
    fatal(err_msg("os::javaTimeMillis: gettimeofday (%s)", strerror(errno)));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1720
  return jlong(t.tv_sec) * 1000  +  jlong(t.tv_usec) / 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1721
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1722
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1723
jlong os::javaTimeNanos() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1724
  return (jlong)getTimeNanos();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1725
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1726
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1727
void os::javaTimeNanos_info(jvmtiTimerInfo *info_ptr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1728
  info_ptr->max_value = ALL_64_BITS;      // gethrtime() uses all 64 bits
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1729
  info_ptr->may_skip_backward = false;    // not subject to resetting or drifting
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1730
  info_ptr->may_skip_forward = false;     // not subject to resetting or drifting
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1731
  info_ptr->kind = JVMTI_TIMER_ELAPSED;   // elapsed not CPU time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1732
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1733
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1734
char * os::local_time_string(char *buf, size_t buflen) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1735
  struct tm t;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1736
  time_t long_time;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1737
  time(&long_time);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1738
  localtime_r(&long_time, &t);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1739
  jio_snprintf(buf, buflen, "%d-%02d-%02d %02d:%02d:%02d",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1740
               t.tm_year + 1900, t.tm_mon + 1, t.tm_mday,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1741
               t.tm_hour, t.tm_min, t.tm_sec);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1742
  return buf;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1743
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1744
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1745
// Note: os::shutdown() might be called very early during initialization, or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1746
// called from signal handler. Before adding something to os::shutdown(), make
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1747
// sure it is async-safe and can handle partially initialized VM.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1748
void os::shutdown() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1749
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1750
  // allow PerfMemory to attempt cleanup of any persistent resources
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1751
  perfMemory_exit();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1752
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1753
  // needs to remove object in file system
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1754
  AttachListener::abort();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1755
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1756
  // flush buffered output, finish log files
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1757
  ostream_abort();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1758
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1759
  // Check for abort hook
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1760
  abort_hook_t abort_hook = Arguments::abort_hook();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1761
  if (abort_hook != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1762
    abort_hook();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1763
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1764
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1765
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1766
// Note: os::abort() might be called very early during initialization, or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1767
// called from signal handler. Before adding something to os::abort(), make
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1768
// sure it is async-safe and can handle partially initialized VM.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1769
void os::abort(bool dump_core) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1770
  os::shutdown();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1771
  if (dump_core) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1772
#ifndef PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1773
    fdStream out(defaultStream::output_fd());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1774
    out.print_raw("Current thread is ");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1775
    char buf[16];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1776
    jio_snprintf(buf, sizeof(buf), UINTX_FORMAT, os::current_thread_id());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1777
    out.print_raw_cr(buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1778
    out.print_raw_cr("Dumping core ...");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1779
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1780
    ::abort(); // dump core (for debugging)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1781
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1782
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1783
  ::exit(1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1784
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1785
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1786
// Die immediately, no exit hook, no abort hook, no cleanup.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1787
void os::die() {
15742
b0ec3b170702 8007779: os::die() on solaris should generate core file
sla
parents: 15234
diff changeset
  1788
  ::abort(); // dump core (for debugging)
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1789
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1790
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1791
// unused
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1792
void os::set_error_file(const char *logfile) {}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1793
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1794
// DLL functions
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1795
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1796
const char* os::dll_file_extension() { return ".so"; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1797
7901
ea3d83447861 7009828: Fix for 6938627 breaks visualvm monitoring when -Djava.io.tmpdir is defined
coleenp
parents: 7726
diff changeset
  1798
// This must be hard coded because it's the system's temporary
ea3d83447861 7009828: Fix for 6938627 breaks visualvm monitoring when -Djava.io.tmpdir is defined
coleenp
parents: 7726
diff changeset
  1799
// directory not the java application's temp directory, ala java.io.tmpdir.
ea3d83447861 7009828: Fix for 6938627 breaks visualvm monitoring when -Djava.io.tmpdir is defined
coleenp
parents: 7726
diff changeset
  1800
const char* os::get_temp_directory() { return "/tmp"; }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1801
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1802
static bool file_exists(const char* filename) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1803
  struct stat statbuf;
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1804
  if (filename == NULL || strlen(filename) == 0) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1805
    return false;
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1806
  }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1807
  return os::stat(filename, &statbuf) == 0;
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1808
}
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1809
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1810
bool os::dll_build_name(char* buffer, size_t buflen,
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1811
                        const char* pname, const char* fname) {
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1812
  bool retval = false;
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1813
  const size_t pnamelen = pname ? strlen(pname) : 0;
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1814
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1815
  // Return error on buffer overflow.
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1816
  if (pnamelen + strlen(fname) + 10 > (size_t) buflen) {
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1817
    return retval;
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1818
  }
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1819
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1820
  if (pnamelen == 0) {
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1821
    snprintf(buffer, buflen, "lib%s.so", fname);
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1822
    retval = true;
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1823
  } else if (strchr(pname, *os::path_separator()) != NULL) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1824
    int n;
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1825
    char** pelements = split_path(pname, &n);
16669
fb3397cee116 8006103: [parfait] Possible null pointer dereference at hotspot/src/os/linux/vm/os_linux.cpp; os_windows.cpp; os_solaris.cpp; os_bsd.cpp
ccheung
parents: 15960
diff changeset
  1826
    if (pelements == NULL) {
16672
dcubed
parents: 16605 16670
diff changeset
  1827
      return false;
16669
fb3397cee116 8006103: [parfait] Possible null pointer dereference at hotspot/src/os/linux/vm/os_linux.cpp; os_windows.cpp; os_solaris.cpp; os_bsd.cpp
ccheung
parents: 15960
diff changeset
  1828
    }
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1829
    for (int i = 0 ; i < n ; i++) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1830
      // really shouldn't be NULL but what the heck, check can't hurt
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1831
      if (pelements[i] == NULL || strlen(pelements[i]) == 0) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1832
        continue; // skip the empty path values
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1833
      }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1834
      snprintf(buffer, buflen, "%s/lib%s.so", pelements[i], fname);
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1835
      if (file_exists(buffer)) {
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1836
        retval = true;
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1837
        break;
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1838
      }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1839
    }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1840
    // release the storage
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1841
    for (int i = 0 ; i < n ; i++) {
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1842
      if (pelements[i] != NULL) {
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  1843
        FREE_C_HEAP_ARRAY(char, pelements[i], mtInternal);
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1844
      }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1845
    }
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1846
    if (pelements != NULL) {
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  1847
      FREE_C_HEAP_ARRAY(char*, pelements, mtInternal);
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1848
    }
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1849
  } else {
2358
7c8346929fc6 6819213: revive sun.boot.library.path
phh
parents: 2268
diff changeset
  1850
    snprintf(buffer, buflen, "%s/lib%s.so", pname, fname);
14471
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1851
    retval = true;
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1852
  }
f3a6b82e25cf 8001185: parsing of sun.boot.library.path in os::dll_build_name somewhat broken
bpittore
parents: 14120
diff changeset
  1853
  return retval;
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1854
}
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  1855
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
  1856
// check if addr is inside libjvm.so
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1857
bool os::address_is_in_vm(address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1858
  static address libjvm_base_addr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1859
  Dl_info dlinfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1860
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1861
  if (libjvm_base_addr == NULL) {
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1862
    if (dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo) != 0) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1863
      libjvm_base_addr = (address)dlinfo.dli_fbase;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1864
    }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1865
    assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1866
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1867
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1868
  if (dladdr((void *)addr, &dlinfo) != 0) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1869
    if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1870
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1871
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1872
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1873
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1874
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1875
typedef int (*dladdr1_func_type) (void *, Dl_info *, void **, int);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1876
static dladdr1_func_type dladdr1_func = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1877
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1878
bool os::dll_address_to_function_name(address addr, char *buf,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1879
                                      int buflen, int * offset) {
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1880
  // buf is not optional, but offset is optional
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1881
  assert(buf != NULL, "sanity check");
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1882
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1883
  Dl_info dlinfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1884
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1885
  // dladdr1_func was initialized in os::init()
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1886
  if (dladdr1_func != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1887
    // yes, we have dladdr1
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1888
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1889
    // Support for dladdr1 is checked at runtime; it may be
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1890
    // available even if the vm is built on a machine that does
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1891
    // not have dladdr1 support.  Make sure there is a value for
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1892
    // RTLD_DL_SYMENT.
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1893
    #ifndef RTLD_DL_SYMENT
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1894
    #define RTLD_DL_SYMENT 1
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1895
    #endif
8329
96eacc5e391f 7018101: os::dll_address_to_function_name returning wrong answers in 64 bit
never
parents: 7901
diff changeset
  1896
#ifdef _LP64
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1897
    Elf64_Sym * info;
8329
96eacc5e391f 7018101: os::dll_address_to_function_name returning wrong answers in 64 bit
never
parents: 7901
diff changeset
  1898
#else
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1899
    Elf32_Sym * info;
8329
96eacc5e391f 7018101: os::dll_address_to_function_name returning wrong answers in 64 bit
never
parents: 7901
diff changeset
  1900
#endif
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1901
    if (dladdr1_func((void *)addr, &dlinfo, (void **)&info,
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1902
                     RTLD_DL_SYMENT) != 0) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1903
      // see if we have a matching symbol that covers our address
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1904
      if (dlinfo.dli_saddr != NULL &&
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1905
          (char *)dlinfo.dli_saddr + info->st_size > (char *)addr) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1906
        if (dlinfo.dli_sname != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1907
          if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1908
            jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1909
          }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1910
          if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
7447
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
  1911
          return true;
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
  1912
        }
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
  1913
      }
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1914
      // no matching symbol so try for just file info
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1915
      if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
7447
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
  1916
        if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1917
                            buf, buflen, offset, dlinfo.dli_fname)) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1918
          return true;
7447
32c42d627f41 7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents: 7397
diff changeset
  1919
        }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1920
      }
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1921
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1922
    buf[0] = '\0';
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1923
    if (offset != NULL) *offset  = -1;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1924
    return false;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1925
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1926
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1927
  // no, only dladdr is available
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1928
  if (dladdr((void *)addr, &dlinfo) != 0) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1929
    // see if we have a matching symbol
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1930
    if (dlinfo.dli_saddr != NULL && dlinfo.dli_sname != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1931
      if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1932
        jio_snprintf(buf, buflen, dlinfo.dli_sname);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1933
      }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1934
      if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1935
      return true;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1936
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1937
    // no matching symbol so try for just file info
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1938
    if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1939
      if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1940
                          buf, buflen, offset, dlinfo.dli_fname)) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1941
        return true;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1942
      }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1943
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1944
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1945
  buf[0] = '\0';
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1946
  if (offset != NULL) *offset  = -1;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1947
  return false;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1948
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1949
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1950
bool os::dll_address_to_library_name(address addr, char* buf,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1951
                                     int buflen, int* offset) {
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1952
  // buf is not optional, but offset is optional
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1953
  assert(buf != NULL, "sanity check");
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1954
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1955
  Dl_info dlinfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1956
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1957
  if (dladdr((void*)addr, &dlinfo) != 0) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1958
    if (dlinfo.dli_fname != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1959
      jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1960
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1961
    if (dlinfo.dli_fbase != NULL && offset != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1962
      *offset = addr - (address)dlinfo.dli_fbase;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1963
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1964
    return true;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1965
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1966
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1967
  buf[0] = '\0';
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1968
  if (offset) *offset = -1;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1969
  return false;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1970
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1971
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1972
// Prints the names and full paths of all opened dynamic libraries
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1973
// for current process
489c9b5090e2 Initial load
duke
parents:
diff changeset
  1974
void os::print_dll_info(outputStream * st) {
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1975
  Dl_info dli;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1976
  void *handle;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1977
  Link_map *map;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1978
  Link_map *p;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1979
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1980
  st->print_cr("Dynamic libraries:"); st->flush();
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1981
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1982
  if (dladdr(CAST_FROM_FN_PTR(void *, os::print_dll_info), &dli) == 0 ||
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1983
      dli.dli_fname == NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1984
    st->print_cr("Error: Cannot print dynamic libraries.");
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1985
    return;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1986
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1987
  handle = dlopen(dli.dli_fname, RTLD_LAZY);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1988
  if (handle == NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1989
    st->print_cr("Error: Cannot print dynamic libraries.");
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1990
    return;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1991
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1992
  dlinfo(handle, RTLD_DI_LINKMAP, &map);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1993
  if (map == NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1994
    st->print_cr("Error: Cannot print dynamic libraries.");
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1995
    return;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1996
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1997
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1998
  while (map->l_prev != NULL)
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  1999
    map = map->l_prev;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2000
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2001
  while (map != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2002
    st->print_cr(PTR_FORMAT " \t%s", map->l_addr, map->l_name);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2003
    map = map->l_next;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2004
  }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2005
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2006
  dlclose(handle);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2007
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2008
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2009
  // Loads .dll/.so and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2010
  // in case of error it checks if .dll/.so was built for the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2011
  // same architecture as Hotspot is running on
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2012
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2013
void * os::dll_load(const char *filename, char *ebuf, int ebuflen)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2014
{
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2015
  void * result= ::dlopen(filename, RTLD_LAZY);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2016
  if (result != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2017
    // Successful loading
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2018
    return result;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2019
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2020
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2021
  Elf32_Ehdr elf_head;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2022
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2023
  // Read system error message into ebuf
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2024
  // It may or may not be overwritten below
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2025
  ::strncpy(ebuf, ::dlerror(), ebuflen-1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2026
  ebuf[ebuflen-1]='\0';
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2027
  int diag_msg_max_length=ebuflen-strlen(ebuf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2028
  char* diag_msg_buf=ebuf+strlen(ebuf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2029
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2030
  if (diag_msg_max_length==0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2031
    // No more space in ebuf for additional diagnostics message
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2032
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2033
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2034
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2035
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2036
  int file_descriptor= ::open(filename, O_RDONLY | O_NONBLOCK);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2037
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2038
  if (file_descriptor < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2039
    // Can't open library, report dlerror() message
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2040
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2041
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2042
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2043
  bool failed_to_read_elf_head=
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2044
    (sizeof(elf_head)!=
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2045
        (::read(file_descriptor, &elf_head,sizeof(elf_head)))) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2046
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2047
  ::close(file_descriptor);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2048
  if (failed_to_read_elf_head) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2049
    // file i/o error - report dlerror() msg
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2050
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2051
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2052
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2053
  typedef struct {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2054
    Elf32_Half  code;         // Actual value as defined in elf.h
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2055
    Elf32_Half  compat_class; // Compatibility of archs at VM's sense
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2056
    char        elf_class;    // 32 or 64 bit
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2057
    char        endianess;    // MSB or LSB
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2058
    char*       name;         // String representation
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2059
  } arch_t;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2060
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2061
  static const arch_t arch_array[]={
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2062
    {EM_386,         EM_386,     ELFCLASS32, ELFDATA2LSB, (char*)"IA 32"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2063
    {EM_486,         EM_386,     ELFCLASS32, ELFDATA2LSB, (char*)"IA 32"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2064
    {EM_IA_64,       EM_IA_64,   ELFCLASS64, ELFDATA2LSB, (char*)"IA 64"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2065
    {EM_X86_64,      EM_X86_64,  ELFCLASS64, ELFDATA2LSB, (char*)"AMD 64"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2066
    {EM_SPARC,       EM_SPARC,   ELFCLASS32, ELFDATA2MSB, (char*)"Sparc 32"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2067
    {EM_SPARC32PLUS, EM_SPARC,   ELFCLASS32, ELFDATA2MSB, (char*)"Sparc 32"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2068
    {EM_SPARCV9,     EM_SPARCV9, ELFCLASS64, ELFDATA2MSB, (char*)"Sparc v9 64"},
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2069
    {EM_PPC,         EM_PPC,     ELFCLASS32, ELFDATA2MSB, (char*)"Power PC 32"},
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  2070
    {EM_PPC64,       EM_PPC64,   ELFCLASS64, ELFDATA2MSB, (char*)"Power PC 64"},
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  2071
    {EM_ARM,         EM_ARM,     ELFCLASS32, ELFDATA2LSB, (char*)"ARM 32"}
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2072
  };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2073
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2074
  #if  (defined IA32)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2075
    static  Elf32_Half running_arch_code=EM_386;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2076
  #elif   (defined AMD64)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2077
    static  Elf32_Half running_arch_code=EM_X86_64;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2078
  #elif  (defined IA64)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2079
    static  Elf32_Half running_arch_code=EM_IA_64;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2080
  #elif  (defined __sparc) && (defined _LP64)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2081
    static  Elf32_Half running_arch_code=EM_SPARCV9;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2082
  #elif  (defined __sparc) && (!defined _LP64)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2083
    static  Elf32_Half running_arch_code=EM_SPARC;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2084
  #elif  (defined __powerpc64__)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2085
    static  Elf32_Half running_arch_code=EM_PPC64;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2086
  #elif  (defined __powerpc__)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2087
    static  Elf32_Half running_arch_code=EM_PPC;
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  2088
  #elif (defined ARM)
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  2089
    static  Elf32_Half running_arch_code=EM_ARM;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2090
  #else
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2091
    #error Method os::dll_load requires that one of following is defined:\
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  2092
         IA32, AMD64, IA64, __sparc, __powerpc__, ARM, ARM
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2093
  #endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2094
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2095
  // Identify compatability class for VM's architecture and library's architecture
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2096
  // Obtain string descriptions for architectures
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2097
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2098
  arch_t lib_arch={elf_head.e_machine,0,elf_head.e_ident[EI_CLASS], elf_head.e_ident[EI_DATA], NULL};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2099
  int running_arch_index=-1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2100
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2101
  for (unsigned int i=0 ; i < ARRAY_SIZE(arch_array) ; i++ ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2102
    if (running_arch_code == arch_array[i].code) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2103
      running_arch_index    = i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2104
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2105
    if (lib_arch.code == arch_array[i].code) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2106
      lib_arch.compat_class = arch_array[i].compat_class;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2107
      lib_arch.name         = arch_array[i].name;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2108
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2109
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2110
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2111
  assert(running_arch_index != -1,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2112
    "Didn't find running architecture code (running_arch_code) in arch_array");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2113
  if (running_arch_index == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2114
    // Even though running architecture detection failed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2115
    // we may still continue with reporting dlerror() message
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2116
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2117
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2118
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2119
  if (lib_arch.endianess != arch_array[running_arch_index].endianess) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2120
    ::snprintf(diag_msg_buf, diag_msg_max_length-1," (Possible cause: endianness mismatch)");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2121
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2122
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2123
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2124
  if (lib_arch.elf_class != arch_array[running_arch_index].elf_class) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2125
    ::snprintf(diag_msg_buf, diag_msg_max_length-1," (Possible cause: architecture word width mismatch)");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2126
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2127
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2128
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2129
  if (lib_arch.compat_class != arch_array[running_arch_index].compat_class) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2130
    if ( lib_arch.name!=NULL ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2131
      ::snprintf(diag_msg_buf, diag_msg_max_length-1,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2132
        " (Possible cause: can't load %s-bit .so on a %s-bit platform)",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2133
        lib_arch.name, arch_array[running_arch_index].name);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2134
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2135
      ::snprintf(diag_msg_buf, diag_msg_max_length-1,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2136
      " (Possible cause: can't load this .so (machine code=0x%x) on a %s-bit platform)",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2137
        lib_arch.code,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2138
        arch_array[running_arch_index].name);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2139
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2140
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2141
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2142
  return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2143
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2144
950
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  2145
void* os::dll_lookup(void* handle, const char* name) {
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  2146
  return dlsym(handle, name);
6112b627bb36 6721093: -XX:AppendRatio=N not supported
kamg
parents: 823
diff changeset
  2147
}
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2148
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2149
int os::stat(const char *path, struct stat *sbuf) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2150
  char pathbuf[MAX_PATH];
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2151
  if (strlen(path) > MAX_PATH - 1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2152
    errno = ENAMETOOLONG;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2153
    return -1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2154
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2155
  os::native_path(strcpy(pathbuf, path));
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2156
  return ::stat(pathbuf, sbuf);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2157
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2158
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2159
static bool _print_ascii_file(const char* filename, outputStream* st) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2160
  int fd = ::open(filename, O_RDONLY);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2161
  if (fd == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2162
     return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2163
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2164
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2165
  char buf[32];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2166
  int bytes;
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2167
  while ((bytes = ::read(fd, buf, sizeof(buf))) > 0) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2168
    st->print_raw(buf, bytes);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2169
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2170
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2171
  ::close(fd);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2172
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2173
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2174
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2175
12735
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2176
void os::print_os_info_brief(outputStream* st) {
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2177
  os::Solaris::print_distro_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2178
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2179
  os::Posix::print_uname_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2180
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2181
  os::Solaris::print_libversion_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2182
}
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2183
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2184
void os::print_os_info(outputStream* st) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2185
  st->print("OS:");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2186
12735
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2187
  os::Solaris::print_distro_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2188
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2189
  os::Posix::print_uname_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2190
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2191
  os::Solaris::print_libversion_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2192
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2193
  os::Posix::print_rlimit_info(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2194
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2195
  os::Posix::print_load_average(st);
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2196
}
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2197
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2198
void os::Solaris::print_distro_info(outputStream* st) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2199
  if (!_print_ascii_file("/etc/release", st)) {
12735
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2200
      st->print("Solaris");
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2201
    }
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2202
    st->cr();
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2203
}
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2204
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2205
void os::Solaris::print_libversion_info(outputStream* st) {
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2206
  if (os::Solaris::T2_libthread()) {
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2207
    st->print("  (T2 libthread)");
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2208
  }
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2209
  else {
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2210
    st->print("  (T1 libthread)");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2211
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2212
  st->cr();
12735
3e2e491f4f69 7165755: OS Information much longer on linux than other platforms
nloodin
parents: 11961
diff changeset
  2213
}
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2214
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2215
static bool check_addr0(outputStream* st) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2216
  jboolean status = false;
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2217
  int fd = ::open("/proc/self/map",O_RDONLY);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2218
  if (fd >= 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2219
    prmap_t p;
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2220
    while(::read(fd, &p, sizeof(p)) > 0) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2221
      if (p.pr_vaddr == 0x0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2222
        st->print("Warning: Address: 0x%x, Size: %dK, ",p.pr_vaddr, p.pr_size/1024, p.pr_mapname);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2223
        st->print("Mapped file: %s, ", p.pr_mapname[0] == '\0' ? "None" : p.pr_mapname);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2224
        st->print("Access:");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2225
        st->print("%s",(p.pr_mflags & MA_READ)  ? "r" : "-");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2226
        st->print("%s",(p.pr_mflags & MA_WRITE) ? "w" : "-");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2227
        st->print("%s",(p.pr_mflags & MA_EXEC)  ? "x" : "-");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2228
        st->cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2229
        status = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2230
      }
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2231
      ::close(fd);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2232
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2233
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2234
  return status;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2235
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2236
10023
e99d9a03c0f5 7061225: os::print_cpu_info() should support os-specific data
jcoomes
parents: 9419
diff changeset
  2237
void os::pd_print_cpu_info(outputStream* st) {
e99d9a03c0f5 7061225: os::print_cpu_info() should support os-specific data
jcoomes
parents: 9419
diff changeset
  2238
  // Nothing to do for now.
e99d9a03c0f5 7061225: os::print_cpu_info() should support os-specific data
jcoomes
parents: 9419
diff changeset
  2239
}
e99d9a03c0f5 7061225: os::print_cpu_info() should support os-specific data
jcoomes
parents: 9419
diff changeset
  2240
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2241
void os::print_memory_info(outputStream* st) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2242
  st->print("Memory:");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2243
  st->print(" %dk page", os::vm_page_size()>>10);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2244
  st->print(", physical " UINT64_FORMAT "k", os::physical_memory()>>10);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2245
  st->print("(" UINT64_FORMAT "k free)", os::available_memory() >> 10);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2246
  st->cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2247
  (void) check_addr0(st);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2248
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2249
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2250
// Taken from /usr/include/sys/machsig.h  Supposed to be architecture specific
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2251
// but they're the same for all the solaris architectures that we support.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2252
const char *ill_names[] = { "ILL0", "ILL_ILLOPC", "ILL_ILLOPN", "ILL_ILLADR",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2253
                          "ILL_ILLTRP", "ILL_PRVOPC", "ILL_PRVREG",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2254
                          "ILL_COPROC", "ILL_BADSTK" };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2255
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2256
const char *fpe_names[] = { "FPE0", "FPE_INTDIV", "FPE_INTOVF", "FPE_FLTDIV",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2257
                          "FPE_FLTOVF", "FPE_FLTUND", "FPE_FLTRES",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2258
                          "FPE_FLTINV", "FPE_FLTSUB" };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2259
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2260
const char *segv_names[] = { "SEGV0", "SEGV_MAPERR", "SEGV_ACCERR" };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2261
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2262
const char *bus_names[] = { "BUS0", "BUS_ADRALN", "BUS_ADRERR", "BUS_OBJERR" };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2263
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2264
void os::print_siginfo(outputStream* st, void* siginfo) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2265
  st->print("siginfo:");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2266
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2267
  const int buflen = 100;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2268
  char buf[buflen];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2269
  siginfo_t *si = (siginfo_t*)siginfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2270
  st->print("si_signo=%s: ", os::exception_name(si->si_signo, buf, buflen));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2271
  char *err = strerror(si->si_errno);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2272
  if (si->si_errno != 0 && err != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2273
    st->print("si_errno=%s", err);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2274
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2275
    st->print("si_errno=%d", si->si_errno);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2276
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2277
  const int c = si->si_code;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2278
  assert(c > 0, "unexpected si_code");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2279
  switch (si->si_signo) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2280
  case SIGILL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2281
    st->print(", si_code=%d (%s)", c, c > 8 ? "" : ill_names[c]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2282
    st->print(", si_addr=" PTR_FORMAT, si->si_addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2283
    break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2284
  case SIGFPE:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2285
    st->print(", si_code=%d (%s)", c, c > 9 ? "" : fpe_names[c]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2286
    st->print(", si_addr=" PTR_FORMAT, si->si_addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2287
    break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2288
  case SIGSEGV:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2289
    st->print(", si_code=%d (%s)", c, c > 2 ? "" : segv_names[c]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2290
    st->print(", si_addr=" PTR_FORMAT, si->si_addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2291
    break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2292
  case SIGBUS:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2293
    st->print(", si_code=%d (%s)", c, c > 3 ? "" : bus_names[c]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2294
    st->print(", si_addr=" PTR_FORMAT, si->si_addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2295
    break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2296
  default:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2297
    st->print(", si_code=%d", si->si_code);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2298
    // no si_addr
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2299
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2300
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2301
  if ((si->si_signo == SIGBUS || si->si_signo == SIGSEGV) &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2302
      UseSharedSpaces) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2303
    FileMapInfo* mapinfo = FileMapInfo::current_info();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2304
    if (mapinfo->is_in_shared_space(si->si_addr)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2305
      st->print("\n\nError accessing class data sharing archive."   \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2306
                " Mapped file inaccessible during execution, "      \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2307
                " possible disk/network problem.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2308
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2309
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2310
  st->cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2311
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2312
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2313
// Moved from whole group, because we need them here for diagnostic
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2314
// prints.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2315
#define OLDMAXSIGNUM 32
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2316
static int Maxsignum = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2317
static int *ourSigFlags = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2318
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2319
extern "C" void sigINTRHandler(int, siginfo_t*, void*);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2320
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2321
int os::Solaris::get_our_sigflags(int sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2322
  assert(ourSigFlags!=NULL, "signal data structure not initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2323
  assert(sig > 0 && sig < Maxsignum, "vm signal out of expected range");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2324
  return ourSigFlags[sig];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2325
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2326
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2327
void os::Solaris::set_our_sigflags(int sig, int flags) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2328
  assert(ourSigFlags!=NULL, "signal data structure not initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2329
  assert(sig > 0 && sig < Maxsignum, "vm signal out of expected range");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2330
  ourSigFlags[sig] = flags;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2331
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2332
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2333
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2334
static const char* get_signal_handler_name(address handler,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2335
                                           char* buf, int buflen) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2336
  int offset;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2337
  bool found = os::dll_address_to_library_name(handler, buf, buflen, &offset);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2338
  if (found) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2339
    // skip directory names
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2340
    const char *p1, *p2;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2341
    p1 = buf;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2342
    size_t len = strlen(os::file_separator());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2343
    while ((p2 = strstr(p1, os::file_separator())) != NULL) p1 = p2 + len;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2344
    jio_snprintf(buf, buflen, "%s+0x%x", p1, offset);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2345
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2346
    jio_snprintf(buf, buflen, PTR_FORMAT, handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2347
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2348
  return buf;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2349
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2350
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2351
static void print_signal_handler(outputStream* st, int sig,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2352
                                  char* buf, size_t buflen) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2353
  struct sigaction sa;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2354
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2355
  sigaction(sig, NULL, &sa);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2356
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2357
  st->print("%s: ", os::exception_name(sig, buf, buflen));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2358
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2359
  address handler = (sa.sa_flags & SA_SIGINFO)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2360
                  ? CAST_FROM_FN_PTR(address, sa.sa_sigaction)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2361
                  : CAST_FROM_FN_PTR(address, sa.sa_handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2362
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2363
  if (handler == CAST_FROM_FN_PTR(address, SIG_DFL)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2364
    st->print("SIG_DFL");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2365
  } else if (handler == CAST_FROM_FN_PTR(address, SIG_IGN)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2366
    st->print("SIG_IGN");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2367
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2368
    st->print("[%s]", get_signal_handler_name(handler, buf, buflen));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2369
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2370
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2371
  st->print(", sa_mask[0]=" PTR32_FORMAT, *(uint32_t*)&sa.sa_mask);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2372
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2373
  address rh = VMError::get_resetted_sighandler(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2374
  // May be, handler was resetted by VMError?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2375
  if(rh != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2376
    handler = rh;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2377
    sa.sa_flags = VMError::get_resetted_sigflags(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2378
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2379
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2380
  st->print(", sa_flags="   PTR32_FORMAT, sa.sa_flags);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2381
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2382
  // Check: is it our handler?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2383
  if(handler == CAST_FROM_FN_PTR(address, signalHandler) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2384
     handler == CAST_FROM_FN_PTR(address, sigINTRHandler)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2385
    // It is our signal handler
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2386
    // check for flags
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2387
    if(sa.sa_flags != os::Solaris::get_our_sigflags(sig)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2388
      st->print(
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2389
        ", flags was changed from " PTR32_FORMAT ", consider using jsig library",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2390
        os::Solaris::get_our_sigflags(sig));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2391
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2392
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2393
  st->cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2394
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2395
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2396
void os::print_signal_handlers(outputStream* st, char* buf, size_t buflen) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2397
  st->print_cr("Signal Handlers:");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2398
  print_signal_handler(st, SIGSEGV, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2399
  print_signal_handler(st, SIGBUS , buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2400
  print_signal_handler(st, SIGFPE , buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2401
  print_signal_handler(st, SIGPIPE, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2402
  print_signal_handler(st, SIGXFSZ, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2403
  print_signal_handler(st, SIGILL , buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2404
  print_signal_handler(st, INTERRUPT_SIGNAL, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2405
  print_signal_handler(st, ASYNC_SIGNAL, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2406
  print_signal_handler(st, BREAK_SIGNAL, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2407
  print_signal_handler(st, SHUTDOWN1_SIGNAL , buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2408
  print_signal_handler(st, SHUTDOWN2_SIGNAL , buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2409
  print_signal_handler(st, SHUTDOWN3_SIGNAL, buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2410
  print_signal_handler(st, os::Solaris::SIGinterrupt(), buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2411
  print_signal_handler(st, os::Solaris::SIGasync(), buf, buflen);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2412
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2413
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2414
static char saved_jvm_path[MAXPATHLEN] = { 0 };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2415
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
  2416
// Find the full path to the current module, libjvm.so
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2417
void os::jvm_path(char *buf, jint buflen) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2418
  // Error checking.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2419
  if (buflen < MAXPATHLEN) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2420
    assert(false, "must use a large-enough buffer");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2421
    buf[0] = '\0';
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2422
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2423
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2424
  // Lazy resolve the path to current module.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2425
  if (saved_jvm_path[0] != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2426
    strcpy(buf, saved_jvm_path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2427
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2428
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2429
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2430
  Dl_info dlinfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2431
  int ret = dladdr(CAST_FROM_FN_PTR(void *, os::jvm_path), &dlinfo);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2432
  assert(ret != 0, "cannot locate libjvm");
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2433
  if (ret != 0 && dlinfo.dli_fname != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2434
    realpath((char *)dlinfo.dli_fname, buf);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2435
  } else {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2436
    buf[0] = '\0';
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2437
    return;
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  2438
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2439
8476
7e34c2d4cf9b 7022037: Pause when exiting if debugger is attached on windows
sla
parents: 8106
diff changeset
  2440
  if (Arguments::created_by_gamma_launcher()) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2441
    // Support for the gamma launcher.  Typical value for buf is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2442
    // "<JAVA_HOME>/jre/lib/<arch>/<vmtype>/libjvm.so".  If "/jre/lib/" appears at
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2443
    // the right place in the string, then assume we are installed in a JDK and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2444
    // we're done.  Otherwise, check for a JAVA_HOME environment variable and fix
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2445
    // up the path so it looks like libjvm.so is installed there (append a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2446
    // fake suffix hotspot/libjvm.so).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2447
    const char *p = buf + strlen(buf) - 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2448
    for (int count = 0; p > buf && count < 5; ++count) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2449
      for (--p; p > buf && *p != '/'; --p)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2450
        /* empty */ ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2451
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2452
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2453
    if (strncmp(p, "/jre/lib/", 9) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2454
      // Look for JAVA_HOME in the environment.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2455
      char* java_home_var = ::getenv("JAVA_HOME");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2456
      if (java_home_var != NULL && java_home_var[0] != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2457
        char cpu_arch[12];
5922
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2458
        char* jrelib_p;
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2459
        int   len;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2460
        sysinfo(SI_ARCHITECTURE, cpu_arch, sizeof(cpu_arch));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2461
#ifdef _LP64
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2462
        // If we are on sparc running a 64-bit vm, look in jre/lib/sparcv9.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2463
        if (strcmp(cpu_arch, "sparc") == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2464
          strcat(cpu_arch, "v9");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2465
        } else if (strcmp(cpu_arch, "i386") == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2466
          strcpy(cpu_arch, "amd64");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2467
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2468
#endif
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
  2469
        // Check the current module name "libjvm.so".
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2470
        p = strrchr(buf, '/');
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2471
        assert(strstr(p, "/libjvm") == p, "invalid library name");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2472
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2473
        realpath(java_home_var, buf);
5922
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2474
        // determine if this is a legacy image or modules image
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2475
        // modules image doesn't have "jre" subdirectory
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2476
        len = strlen(buf);
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2477
        jrelib_p = buf + len;
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2478
        snprintf(jrelib_p, buflen-len, "/jre/lib/%s", cpu_arch);
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2479
        if (0 != access(buf, F_OK)) {
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2480
          snprintf(jrelib_p, buflen-len, "/lib/%s", cpu_arch);
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2481
        }
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2482
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2483
        if (0 == access(buf, F_OK)) {
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
  2484
          // Use current module name "libjvm.so"
5922
883ecb03d008 6967423: Hotspot support for modules image
mchung
parents: 5547
diff changeset
  2485
          len = strlen(buf);
15096
3db45569f8c0 8005044: remove crufty '_g' support from HS runtime code
dcubed
parents: 14633
diff changeset
  2486
          snprintf(buf + len, buflen-len, "/hotspot/libjvm.so");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2487
        } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2488
          // Go back to path of .so
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2489
          realpath((char *)dlinfo.dli_fname, buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2490
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2491
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2492
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2493
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2494
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2495
  strcpy(saved_jvm_path, buf);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2496
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2497
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2498
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2499
void os::print_jni_name_prefix_on(outputStream* st, int args_size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2500
  // no prefix required, not even "_"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2501
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2502
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2503
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2504
void os::print_jni_name_suffix_on(outputStream* st, int args_size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2505
  // no suffix required
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2506
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2507
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2508
// This method is a copy of JDK's sysGetLastErrorString
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2509
// from src/solaris/hpi/src/system_md.c
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2510
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2511
size_t os::lasterror(char *buf, size_t len) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2512
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2513
  if (errno == 0)  return 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2514
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2515
  const char *s = ::strerror(errno);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2516
  size_t n = ::strlen(s);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2517
  if (n >= len) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2518
    n = len - 1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2519
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2520
  ::strncpy(buf, s, n);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2521
  buf[n] = '\0';
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2522
  return n;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2523
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  2524
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2525
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2526
// sun.misc.Signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2527
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2528
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2529
  static void UserHandler(int sig, void *siginfo, void *context) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2530
    // Ctrl-C is pressed during error reporting, likely because the error
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2531
    // handler fails to abort. Let VM die immediately.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2532
    if (sig == SIGINT && is_error_reported()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2533
       os::die();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2534
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2535
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2536
    os::signal_notify(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2537
    // We do not need to reinstate the signal handler each time...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2538
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2539
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2540
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2541
void* os::user_handler() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2542
  return CAST_FROM_FN_PTR(void*, UserHandler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2543
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2544
18025
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2545
class Semaphore : public StackObj {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2546
  public:
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2547
    Semaphore();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2548
    ~Semaphore();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2549
    void signal();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2550
    void wait();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2551
    bool trywait();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2552
    bool timedwait(unsigned int sec, int nsec);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2553
  private:
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2554
    sema_t _semaphore;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2555
};
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2556
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2557
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2558
Semaphore::Semaphore() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2559
  sema_init(&_semaphore, 0, NULL, NULL);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2560
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2561
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2562
Semaphore::~Semaphore() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2563
  sema_destroy(&_semaphore);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2564
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2565
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2566
void Semaphore::signal() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2567
  sema_post(&_semaphore);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2568
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2569
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2570
void Semaphore::wait() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2571
  sema_wait(&_semaphore);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2572
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2573
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2574
bool Semaphore::trywait() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2575
  return sema_trywait(&_semaphore) == 0;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2576
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2577
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2578
bool Semaphore::timedwait(unsigned int sec, int nsec) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2579
  struct timespec ts;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2580
  unpackTime(&ts, false, (sec * NANOSECS_PER_SEC) + nsec);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2581
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2582
  while (1) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2583
    int result = sema_timedwait(&_semaphore, &ts);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2584
    if (result == 0) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2585
      return true;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2586
    } else if (errno == EINTR) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2587
      continue;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2588
    } else if (errno == ETIME) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2589
      return false;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2590
    } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2591
      return false;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2592
    }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2593
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2594
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  2595
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2596
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2597
  typedef void (*sa_handler_t)(int);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2598
  typedef void (*sa_sigaction_t)(int, siginfo_t *, void *);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2599
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2600
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2601
void* os::signal(int signal_number, void* handler) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2602
  struct sigaction sigAct, oldSigAct;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2603
  sigfillset(&(sigAct.sa_mask));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2604
  sigAct.sa_flags = SA_RESTART & ~SA_RESETHAND;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2605
  sigAct.sa_handler = CAST_TO_FN_PTR(sa_handler_t, handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2606
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2607
  if (sigaction(signal_number, &sigAct, &oldSigAct))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2608
    // -1 means registration failed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2609
    return (void *)-1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2610
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2611
  return CAST_FROM_FN_PTR(void*, oldSigAct.sa_handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2612
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2613
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2614
void os::signal_raise(int signal_number) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2615
  raise(signal_number);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2616
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2617
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2618
/*
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2619
 * The following code is moved from os.cpp for making this
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2620
 * code platform specific, which it is by its very nature.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2621
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2622
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2623
// a counter for each possible signal value
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2624
static int Sigexit = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2625
static int Maxlibjsigsigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2626
static jint *pending_signals = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2627
static int *preinstalled_sigs = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2628
static struct sigaction *chainedsigactions = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2629
static sema_t sig_sem;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2630
typedef int (*version_getting_t)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2631
version_getting_t os::Solaris::get_libjsig_version = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2632
static int libjsigversion = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2633
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2634
int os::sigexitnum_pd() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2635
  assert(Sigexit > 0, "signal memory not yet initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2636
  return Sigexit;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2637
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2638
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2639
void os::Solaris::init_signal_mem() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2640
  // Initialize signal structures
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2641
  Maxsignum = SIGRTMAX;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2642
  Sigexit = Maxsignum+1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2643
  assert(Maxsignum >0, "Unable to obtain max signal number");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2644
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2645
  Maxlibjsigsigs = Maxsignum;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2646
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2647
  // pending_signals has one int per signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2648
  // The additional signal is for SIGEXIT - exit signal to signal_thread
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2649
  pending_signals = (jint *)os::malloc(sizeof(jint) * (Sigexit+1), mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2650
  memset(pending_signals, 0, (sizeof(jint) * (Sigexit+1)));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2651
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2652
  if (UseSignalChaining) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2653
     chainedsigactions = (struct sigaction *)malloc(sizeof(struct sigaction)
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2654
       * (Maxsignum + 1), mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2655
     memset(chainedsigactions, 0, (sizeof(struct sigaction) * (Maxsignum + 1)));
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2656
     preinstalled_sigs = (int *)os::malloc(sizeof(int) * (Maxsignum + 1), mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2657
     memset(preinstalled_sigs, 0, (sizeof(int) * (Maxsignum + 1)));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2658
  }
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2659
  ourSigFlags = (int*)malloc(sizeof(int) * (Maxsignum + 1 ), mtInternal);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2660
  memset(ourSigFlags, 0, sizeof(int) * (Maxsignum + 1));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2661
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2662
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2663
void os::signal_init_pd() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2664
  int ret;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2665
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2666
  ret = ::sema_init(&sig_sem, 0, NULL, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2667
  assert(ret == 0, "sema_init() failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2668
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2669
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2670
void os::signal_notify(int signal_number) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2671
  int ret;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2672
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2673
  Atomic::inc(&pending_signals[signal_number]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2674
  ret = ::sema_post(&sig_sem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2675
  assert(ret == 0, "sema_post() failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2676
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2677
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2678
static int check_pending_signals(bool wait_for_signal) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2679
  int ret;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2680
  while (true) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2681
    for (int i = 0; i < Sigexit + 1; i++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2682
      jint n = pending_signals[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2683
      if (n > 0 && n == Atomic::cmpxchg(n - 1, &pending_signals[i], n)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2684
        return i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2685
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2686
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2687
    if (!wait_for_signal) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2688
      return -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2689
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2690
    JavaThread *thread = JavaThread::current();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2691
    ThreadBlockInVM tbivm(thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2692
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2693
    bool threadIsSuspended;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2694
    do {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2695
      thread->set_suspend_equivalent();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2696
      // cleared by handle_special_suspend_equivalent_condition() or java_suspend_self()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2697
      while((ret = ::sema_wait(&sig_sem)) == EINTR)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2698
          ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2699
      assert(ret == 0, "sema_wait() failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2700
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2701
      // were we externally suspended while we were waiting?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2702
      threadIsSuspended = thread->handle_special_suspend_equivalent_condition();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2703
      if (threadIsSuspended) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2704
        //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2705
        // The semaphore has been incremented, but while we were waiting
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2706
        // another thread suspended us. We don't want to continue running
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2707
        // while suspended because that would surprise the thread that
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2708
        // suspended us.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2709
        //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2710
        ret = ::sema_post(&sig_sem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2711
        assert(ret == 0, "sema_post() failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2712
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2713
        thread->java_suspend_self();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2714
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2715
    } while (threadIsSuspended);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2716
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2717
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2718
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2719
int os::signal_lookup() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2720
  return check_pending_signals(false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2721
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2722
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2723
int os::signal_wait() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2724
  return check_pending_signals(true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2725
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2726
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2727
////////////////////////////////////////////////////////////////////////////////
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2728
// Virtual Memory
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2729
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2730
static int page_size = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2731
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2732
// The mmap MAP_ALIGN flag is supported on Solaris 9 and later.  init_2() will
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2733
// clear this var if support is not available.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2734
static bool has_map_align = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2735
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2736
int os::vm_page_size() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2737
  assert(page_size != -1, "must call os::init");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2738
  return page_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2739
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2740
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2741
// Solaris allocates memory by pages.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2742
int os::vm_allocation_granularity() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2743
  assert(page_size != -1, "must call os::init");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2744
  return page_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2745
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2746
18069
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2747
static bool recoverable_mmap_error(int err) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2748
  // See if the error is one we can let the caller handle. This
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2749
  // list of errno values comes from the Solaris mmap(2) man page.
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2750
  switch (err) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2751
  case EBADF:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2752
  case EINVAL:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2753
  case ENOTSUP:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2754
    // let the caller deal with these errors
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2755
    return true;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2756
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2757
  default:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2758
    // Any remaining errors on this OS can cause our reserved mapping
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2759
    // to be lost. That can cause confusion where different data
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2760
    // structures think they have the same memory mapped. The worst
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2761
    // scenario is if both the VM and a library think they have the
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2762
    // same memory mapped.
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2763
    return false;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2764
  }
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2765
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2766
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2767
static void warn_fail_commit_memory(char* addr, size_t bytes, bool exec,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2768
                                    int err) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2769
  warning("INFO: os::commit_memory(" PTR_FORMAT ", " SIZE_FORMAT
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2770
          ", %d) failed; error='%s' (errno=%d)", addr, bytes, exec,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2771
          strerror(err), err);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2772
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2773
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2774
static void warn_fail_commit_memory(char* addr, size_t bytes,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2775
                                    size_t alignment_hint, bool exec,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2776
                                    int err) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2777
  warning("INFO: os::commit_memory(" PTR_FORMAT ", " SIZE_FORMAT
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2778
          ", " SIZE_FORMAT ", %d) failed; error='%s' (errno=%d)", addr, bytes,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2779
          alignment_hint, exec, strerror(err), err);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2780
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2781
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2782
int os::Solaris::commit_memory_impl(char* addr, size_t bytes, bool exec) {
2268
bea8be80ec88 6541756: Reduce executable C-heap
coleenp
parents: 2259
diff changeset
  2783
  int prot = exec ? PROT_READ|PROT_WRITE|PROT_EXEC : PROT_READ|PROT_WRITE;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2784
  size_t size = bytes;
10494
3f347ed8bd3c 7082969: NUMA interleaving
iveresov
parents: 10272
diff changeset
  2785
  char *res = Solaris::mmap_chunk(addr, size, MAP_PRIVATE|MAP_FIXED, prot);
3f347ed8bd3c 7082969: NUMA interleaving
iveresov
parents: 10272
diff changeset
  2786
  if (res != NULL) {
3f347ed8bd3c 7082969: NUMA interleaving
iveresov
parents: 10272
diff changeset
  2787
    if (UseNUMAInterleaving) {
3f347ed8bd3c 7082969: NUMA interleaving
iveresov
parents: 10272
diff changeset
  2788
      numa_make_global(addr, bytes);
3f347ed8bd3c 7082969: NUMA interleaving
iveresov
parents: 10272
diff changeset
  2789
    }
18069
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2790
    return 0;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2791
  }
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2792
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2793
  int err = errno;  // save errno from mmap() call in mmap_chunk()
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2794
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2795
  if (!recoverable_mmap_error(err)) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2796
    warn_fail_commit_memory(addr, bytes, exec, err);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2797
    vm_exit_out_of_memory(bytes, OOM_MMAP_ERROR, "committing reserved memory.");
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2798
  }
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2799
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2800
  return err;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2801
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2802
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2803
bool os::pd_commit_memory(char* addr, size_t bytes, bool exec) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2804
  return Solaris::commit_memory_impl(addr, bytes, exec) == 0;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2805
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2806
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2807
void os::pd_commit_memory_or_exit(char* addr, size_t bytes, bool exec,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2808
                                  const char* mesg) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2809
  assert(mesg != NULL, "mesg must be specified");
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2810
  int err = os::Solaris::commit_memory_impl(addr, bytes, exec);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2811
  if (err != 0) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2812
    // the caller wants all commit errors to exit with the specified mesg:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2813
    warn_fail_commit_memory(addr, bytes, exec, err);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2814
    vm_exit_out_of_memory(bytes, OOM_MMAP_ERROR, mesg);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2815
  }
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2816
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2817
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2818
int os::Solaris::commit_memory_impl(char* addr, size_t bytes,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2819
                                    size_t alignment_hint, bool exec) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2820
  int err = Solaris::commit_memory_impl(addr, bytes, exec);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2821
  if (err == 0) {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  2822
    if (UseLargePages && (alignment_hint > (size_t)vm_page_size())) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2823
      // If the large page size has been set and the VM
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2824
      // is using large pages, use the large page size
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2825
      // if it is smaller than the alignment hint. This is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2826
      // a case where the VM wants to use a larger alignment size
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2827
      // for its own reasons but still want to use large pages
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2828
      // (which is what matters to setting the mpss range.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2829
      size_t page_size = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2830
      if (large_page_size() < alignment_hint) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2831
        assert(UseLargePages, "Expected to be here for large page use only");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2832
        page_size = large_page_size();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2833
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2834
        // If the alignment hint is less than the large page
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2835
        // size, the VM wants a particular alignment (thus the hint)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2836
        // for internal reasons.  Try to set the mpss range using
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2837
        // the alignment_hint.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2838
        page_size = alignment_hint;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2839
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2840
      // Since this is a hint, ignore any failures.
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  2841
      (void)Solaris::setup_large_pages(addr, bytes, page_size);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2842
    }
18069
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2843
  }
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2844
  return err;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2845
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2846
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2847
bool os::pd_commit_memory(char* addr, size_t bytes, size_t alignment_hint,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2848
                          bool exec) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2849
  return Solaris::commit_memory_impl(addr, bytes, alignment_hint, exec) == 0;
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2850
}
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2851
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2852
void os::pd_commit_memory_or_exit(char* addr, size_t bytes,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2853
                                  size_t alignment_hint, bool exec,
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2854
                                  const char* mesg) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2855
  assert(mesg != NULL, "mesg must be specified");
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2856
  int err = os::Solaris::commit_memory_impl(addr, bytes, alignment_hint, exec);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2857
  if (err != 0) {
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2858
    // the caller wants all commit errors to exit with the specified mesg:
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2859
    warn_fail_commit_memory(addr, bytes, alignment_hint, exec, err);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2860
    vm_exit_out_of_memory(bytes, OOM_MMAP_ERROR, mesg);
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2861
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2862
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2863
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2864
// Uncommit the pages in a specified region.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2865
void os::pd_free_memory(char* addr, size_t bytes, size_t alignment_hint) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2866
  if (madvise(addr, bytes, MADV_FREE) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2867
    debug_only(warning("MADV_FREE failed."));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2868
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2869
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2870
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2871
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2872
bool os::pd_create_stack_guard_pages(char* addr, size_t size) {
18069
e6d4971c8650 8013057: assert(_needs_gc || SafepointSynchronize::is_at_safepoint()) failed: only read at safepoint
dcubed
parents: 18025
diff changeset
  2873
  return os::commit_memory(addr, size, !ExecMem);
5085
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2874
}
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2875
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2876
bool os::remove_stack_guard_pages(char* addr, size_t size) {
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2877
  return os::uncommit_memory(addr, size);
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2878
}
4f0c435f8c3c 6929067: Stack guard pages should be removed when thread is detached
coleenp
parents: 4493
diff changeset
  2879
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2880
// Change the page size in a given range.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  2881
void os::pd_realign_memory(char *addr, size_t bytes, size_t alignment_hint) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2882
  assert((intptr_t)addr % alignment_hint == 0, "Address should be aligned.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2883
  assert((intptr_t)(addr + bytes) % alignment_hint == 0, "End should be aligned.");
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  2884
  if (UseLargePages) {
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  2885
    Solaris::setup_large_pages(addr, bytes, alignment_hint);
9341
347fa5cdbd39 7037939: NUMA: Disable adaptive resizing if SHM large pages are used
iveresov
parents: 8478
diff changeset
  2886
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2887
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2888
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2889
// Tell the OS to make the range local to the first-touching LWP
388
bcc631c5bbec 6684395: Port NUMA-aware allocator to linux
iveresov
parents: 336
diff changeset
  2890
void os::numa_make_local(char *addr, size_t bytes, int lgrp_hint) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2891
  assert((intptr_t)addr % os::vm_page_size() == 0, "Address should be page-aligned.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2892
  if (madvise(addr, bytes, MADV_ACCESS_LWP) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2893
    debug_only(warning("MADV_ACCESS_LWP failed."));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2894
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2895
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2896
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2897
// Tell the OS that this range would be accessed from different LWPs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2898
void os::numa_make_global(char *addr, size_t bytes) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2899
  assert((intptr_t)addr % os::vm_page_size() == 0, "Address should be page-aligned.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2900
  if (madvise(addr, bytes, MADV_ACCESS_MANY) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2901
    debug_only(warning("MADV_ACCESS_MANY failed."));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2902
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2903
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2904
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2905
// Get the number of the locality groups.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2906
size_t os::numa_get_groups_num() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2907
  size_t n = Solaris::lgrp_nlgrps(Solaris::lgrp_cookie());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2908
  return n != -1 ? n : 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2909
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2910
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2911
// Get a list of leaf locality groups. A leaf lgroup is group that
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2912
// doesn't have any children. Typical leaf group is a CPU or a CPU/memory
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2913
// board. An LWP is assigned to one of these groups upon creation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2914
size_t os::numa_get_leaf_groups(int *ids, size_t size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2915
   if ((ids[0] = Solaris::lgrp_root(Solaris::lgrp_cookie())) == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2916
     ids[0] = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2917
     return 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2918
   }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2919
   int result_size = 0, top = 1, bottom = 0, cur = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2920
   for (int k = 0; k < size; k++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2921
     int r = Solaris::lgrp_children(Solaris::lgrp_cookie(), ids[cur],
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2922
                                    (Solaris::lgrp_id_t*)&ids[top], size - top);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2923
     if (r == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2924
       ids[0] = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2925
       return 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2926
     }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2927
     if (!r) {
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2928
       // That's a leaf node.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2929
       assert (bottom <= cur, "Sanity check");
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2930
       // Check if the node has memory
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2931
       if (Solaris::lgrp_resources(Solaris::lgrp_cookie(), ids[cur],
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2932
                                   NULL, 0, LGRP_RSRC_MEM) > 0) {
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2933
         ids[bottom++] = ids[cur];
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2934
       }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2935
     }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2936
     top += r;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2937
     cur++;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2938
   }
976
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2939
   if (bottom == 0) {
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2940
     // Handle a situation, when the OS reports no memory available.
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2941
     // Assume UMA architecture.
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2942
     ids[0] = 0;
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2943
     return 1;
241230d48896 6723228: NUMA allocator: assert(lgrp_id != -1, "No lgrp_id set")
iveresov
parents: 975
diff changeset
  2944
   }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2945
   return bottom;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2946
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2947
1374
4c24294029a9 6711316: Open source the Garbage-First garbage collector
ysr
parents: 391
diff changeset
  2948
// Detect the topology change. Typically happens during CPU plugging-unplugging.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2949
bool os::numa_topology_changed() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2950
  int is_stale = Solaris::lgrp_cookie_stale(Solaris::lgrp_cookie());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2951
  if (is_stale != -1 && is_stale) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2952
    Solaris::lgrp_fini(Solaris::lgrp_cookie());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2953
    Solaris::lgrp_cookie_t c = Solaris::lgrp_init(Solaris::LGRP_VIEW_CALLER);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2954
    assert(c != 0, "Failure to initialize LGRP API");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2955
    Solaris::set_lgrp_cookie(c);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2956
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2957
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2958
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2959
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2960
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2961
// Get the group id of the current LWP.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2962
int os::numa_get_group_id() {
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2963
  int lgrp_id = Solaris::lgrp_home(P_LWPID, P_MYID);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2964
  if (lgrp_id == -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2965
    return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2966
  }
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2967
  const int size = os::numa_get_groups_num();
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2968
  int *ids = (int*)alloca(size * sizeof(int));
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2969
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2970
  // Get the ids of all lgroups with memory; r is the count.
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2971
  int r = Solaris::lgrp_resources(Solaris::lgrp_cookie(), lgrp_id,
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2972
                                  (Solaris::lgrp_id_t*)ids, size, LGRP_RSRC_MEM);
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2973
  if (r <= 0) {
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2974
    return 0;
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2975
  }
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  2976
  return ids[os::random() % r];
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2977
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2978
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2979
// Request information about the page.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2980
bool os::get_page_info(char *start, page_info* info) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2981
  const uint_t info_types[] = { MEMINFO_VLGRP, MEMINFO_VPAGESIZE };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2982
  uint64_t addr = (uintptr_t)start;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2983
  uint64_t outdata[2];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2984
  uint_t validity = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2985
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2986
  if (os::Solaris::meminfo(&addr, 1, info_types, 2, outdata, &validity) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2987
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2988
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2989
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2990
  info->size = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2991
  info->lgrp_id = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2992
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2993
  if ((validity & 1) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2994
    if ((validity & 2) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2995
      info->lgrp_id = outdata[0];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2996
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2997
    if ((validity & 4) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2998
      info->size = outdata[1];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  2999
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3000
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3001
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3002
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3003
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3004
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3005
// Scan the pages from start to end until a page different than
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3006
// the one described in the info parameter is encountered.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3007
char *os::scan_pages(char *start, char* end, page_info* page_expected, page_info* page_found) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3008
  const uint_t info_types[] = { MEMINFO_VLGRP, MEMINFO_VPAGESIZE };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3009
  const size_t types = sizeof(info_types) / sizeof(info_types[0]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3010
  uint64_t addrs[MAX_MEMINFO_CNT], outdata[types * MAX_MEMINFO_CNT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3011
  uint_t validity[MAX_MEMINFO_CNT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3012
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3013
  size_t page_size = MAX2((size_t)os::vm_page_size(), page_expected->size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3014
  uint64_t p = (uint64_t)start;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3015
  while (p < (uint64_t)end) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3016
    addrs[0] = p;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3017
    size_t addrs_count = 1;
15955
6d0d8bea2bcc 8004697: SIGSEGV on Solaris sparc with -XX:+UseNUMA
stefank
parents: 15743
diff changeset
  3018
    while (addrs_count < MAX_MEMINFO_CNT && addrs[addrs_count - 1] + page_size < (uint64_t)end) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3019
      addrs[addrs_count] = addrs[addrs_count - 1] + page_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3020
      addrs_count++;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3021
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3022
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3023
    if (os::Solaris::meminfo(addrs, addrs_count, info_types, types, outdata, validity) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3024
      return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3025
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3026
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3027
    size_t i = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3028
    for (; i < addrs_count; i++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3029
      if ((validity[i] & 1) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3030
        if ((validity[i] & 4) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3031
          if (outdata[types * i + 1] != page_expected->size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3032
            break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3033
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3034
        } else
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3035
          if (page_expected->size != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3036
            break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3037
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3038
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3039
        if ((validity[i] & 2) != 0 && page_expected->lgrp_id > 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3040
          if (outdata[types * i] != page_expected->lgrp_id) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3041
            break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3042
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3043
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3044
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3045
        return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3046
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3047
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3048
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3049
    if (i != addrs_count) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3050
      if ((validity[i] & 2) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3051
        page_found->lgrp_id = outdata[types * i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3052
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3053
        page_found->lgrp_id = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3054
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3055
      if ((validity[i] & 4) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3056
        page_found->size = outdata[types * i + 1];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3057
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3058
        page_found->size = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3059
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3060
      return (char*)addrs[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3061
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3062
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3063
    p = addrs[addrs_count - 1] + page_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3064
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3065
  return end;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3066
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3067
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  3068
bool os::pd_uncommit_memory(char* addr, size_t bytes) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3069
  size_t size = bytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3070
  // Map uncommitted pages PROT_NONE so we fail early if we touch an
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3071
  // uncommitted page. Otherwise, the read/write might succeed if we
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3072
  // have enough swap space to back the physical page.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3073
  return
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3074
    NULL != Solaris::mmap_chunk(addr, size,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3075
                                MAP_PRIVATE|MAP_FIXED|MAP_NORESERVE,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3076
                                PROT_NONE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3077
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3078
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3079
char* os::Solaris::mmap_chunk(char *addr, size_t size, int flags, int prot) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3080
  char *b = (char *)mmap(addr, size, prot, flags, os::Solaris::_dev_zero_fd, 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3081
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3082
  if (b == MAP_FAILED) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3083
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3084
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3085
  return b;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3086
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3087
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3088
char* os::Solaris::anon_mmap(char* requested_addr, size_t bytes, size_t alignment_hint, bool fixed) {
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3089
  char* addr = requested_addr;
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3090
  int flags = MAP_PRIVATE | MAP_NORESERVE;
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3091
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3092
  assert(!(fixed && (alignment_hint > 0)), "alignment hint meaningless with fixed mmap");
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3093
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3094
  if (fixed) {
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3095
    flags |= MAP_FIXED;
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3096
  } else if (has_map_align && (alignment_hint > (size_t) vm_page_size())) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3097
    flags |= MAP_ALIGN;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3098
    addr = (char*) alignment_hint;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3099
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3100
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3101
  // Map uncommitted pages PROT_NONE so we fail early if we touch an
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3102
  // uncommitted page. Otherwise, the read/write might succeed if we
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3103
  // have enough swap space to back the physical page.
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3104
  return mmap_chunk(addr, bytes, flags, PROT_NONE);
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3105
}
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3106
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  3107
char* os::pd_reserve_memory(size_t bytes, char* requested_addr, size_t alignment_hint) {
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3108
  char* addr = Solaris::anon_mmap(requested_addr, bytes, alignment_hint, (requested_addr != NULL));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3109
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3110
  guarantee(requested_addr == NULL || requested_addr == addr,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3111
            "OS failed to return requested mmap address.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3112
  return addr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3113
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3114
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3115
// Reserve memory at an arbitrary address, only if that area is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3116
// available (and not reserved for something else).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3117
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  3118
char* os::pd_attempt_reserve_memory_at(size_t bytes, char* requested_addr) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3119
  const int max_tries = 10;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3120
  char* base[max_tries];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3121
  size_t size[max_tries];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3122
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3123
  // Solaris adds a gap between mmap'ed regions.  The size of the gap
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3124
  // is dependent on the requested size and the MMU.  Our initial gap
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3125
  // value here is just a guess and will be corrected later.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3126
  bool had_top_overlap = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3127
  bool have_adjusted_gap = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3128
  size_t gap = 0x400000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3129
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3130
  // Assert only that the size is a multiple of the page size, since
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3131
  // that's all that mmap requires, and since that's all we really know
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3132
  // about at this low abstraction level.  If we need higher alignment,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3133
  // we can either pass an alignment to this method or verify alignment
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3134
  // in one of the methods further up the call chain.  See bug 5044738.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3135
  assert(bytes % os::vm_page_size() == 0, "reserving unexpected size block");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3136
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3137
  // Since snv_84, Solaris attempts to honor the address hint - see 5003415.
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3138
  // Give it a try, if the kernel honors the hint we can return immediately.
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3139
  char* addr = Solaris::anon_mmap(requested_addr, bytes, 0, false);
14120
7d298141c258 7199092: NMT: NMT needs to deal overlapped virtual memory ranges
zgu
parents: 13975
diff changeset
  3140
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3141
  volatile int err = errno;
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3142
  if (addr == requested_addr) {
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3143
    return addr;
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3144
  } else if (addr != NULL) {
14120
7d298141c258 7199092: NMT: NMT needs to deal overlapped virtual memory ranges
zgu
parents: 13975
diff changeset
  3145
    pd_unmap_memory(addr, bytes);
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3146
  }
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3147
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3148
  if (PrintMiscellaneous && Verbose) {
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3149
    char buf[256];
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3150
    buf[0] = '\0';
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3151
    if (addr == NULL) {
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3152
      jio_snprintf(buf, sizeof(buf), ": %s", strerror(err));
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3153
    }
7704
cc9d3ed42704 7006505: Use kstat info to identify SPARC processor
kvn
parents: 7405
diff changeset
  3154
    warning("attempt_reserve_memory_at: couldn't reserve " SIZE_FORMAT " bytes at "
233
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3155
            PTR_FORMAT ": reserve_memory_helper returned " PTR_FORMAT
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3156
            "%s", bytes, requested_addr, addr, buf);
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3157
  }
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3158
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3159
  // Address hint method didn't work.  Fall back to the old method.
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3160
  // In theory, once SNV becomes our oldest supported platform, this
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3161
  // code will no longer be needed.
9f4505c94388 6204603: Modify hotspot to use new Solaris mmap semantics for class data archive file
sbohne
parents: 1
diff changeset
  3162
  //
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3163
  // Repeatedly allocate blocks until the block is allocated at the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3164
  // right spot. Give up after max_tries.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3165
  int i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3166
  for (i = 0; i < max_tries; ++i) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3167
    base[i] = reserve_memory(bytes);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3168
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3169
    if (base[i] != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3170
      // Is this the block we wanted?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3171
      if (base[i] == requested_addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3172
        size[i] = bytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3173
        break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3174
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3175
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3176
      // check that the gap value is right
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3177
      if (had_top_overlap && !have_adjusted_gap) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3178
        size_t actual_gap = base[i-1] - base[i] - bytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3179
        if (gap != actual_gap) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3180
          // adjust the gap value and retry the last 2 allocations
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3181
          assert(i > 0, "gap adjustment code problem");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3182
          have_adjusted_gap = true;  // adjust the gap only once, just in case
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3183
          gap = actual_gap;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3184
          if (PrintMiscellaneous && Verbose) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3185
            warning("attempt_reserve_memory_at: adjusted gap to 0x%lx", gap);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3186
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3187
          unmap_memory(base[i], bytes);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3188
          unmap_memory(base[i-1], size[i-1]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3189
          i-=2;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3190
          continue;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3191
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3192
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3193
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3194
      // Does this overlap the block we wanted? Give back the overlapped
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3195
      // parts and try again.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3196
      //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3197
      // There is still a bug in this code: if top_overlap == bytes,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3198
      // the overlap is offset from requested region by the value of gap.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3199
      // In this case giving back the overlapped part will not work,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3200
      // because we'll give back the entire block at base[i] and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3201
      // therefore the subsequent allocation will not generate a new gap.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3202
      // This could be fixed with a new algorithm that used larger
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3203
      // or variable size chunks to find the requested region -
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3204
      // but such a change would introduce additional complications.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3205
      // It's rare enough that the planets align for this bug,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3206
      // so we'll just wait for a fix for 6204603/5003415 which
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3207
      // will provide a mmap flag to allow us to avoid this business.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3208
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3209
      size_t top_overlap = requested_addr + (bytes + gap) - base[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3210
      if (top_overlap >= 0 && top_overlap < bytes) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3211
        had_top_overlap = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3212
        unmap_memory(base[i], top_overlap);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3213
        base[i] += top_overlap;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3214
        size[i] = bytes - top_overlap;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3215
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3216
        size_t bottom_overlap = base[i] + bytes - requested_addr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3217
        if (bottom_overlap >= 0 && bottom_overlap < bytes) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3218
          if (PrintMiscellaneous && Verbose && bottom_overlap == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3219
            warning("attempt_reserve_memory_at: possible alignment bug");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3220
          }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3221
          unmap_memory(requested_addr, bottom_overlap);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3222
          size[i] = bytes - bottom_overlap;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3223
        } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3224
          size[i] = bytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3225
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3226
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3227
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3228
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3229
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3230
  // Give back the unused reserved pieces.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3231
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3232
  for (int j = 0; j < i; ++j) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3233
    if (base[j] != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3234
      unmap_memory(base[j], size[j]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3235
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3236
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3237
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3238
  return (i < max_tries) ? requested_addr : NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3239
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3240
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  3241
bool os::pd_release_memory(char* addr, size_t bytes) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3242
  size_t size = bytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3243
  return munmap(addr, size) == 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3244
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3245
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3246
static bool solaris_mprotect(char* addr, size_t bytes, int prot) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3247
  assert(addr == (char*)align_size_down((uintptr_t)addr, os::vm_page_size()),
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3248
         "addr must be page aligned");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3249
  int retVal = mprotect(addr, bytes, prot);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3250
  return retVal == 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3251
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3252
823
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3253
// Protect memory (Used to pass readonly pages through
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3254
// JNI GetArray<type>Elements with empty arrays.)
1664
fc9ed50498fb 6727377: VM stack guard pages on Windows should PAGE_READWRITE not PAGE_EXECUTE_READWRITE
coleenp
parents: 1615
diff changeset
  3255
// Also, used for serialization page and for compressed oops null pointer
fc9ed50498fb 6727377: VM stack guard pages on Windows should PAGE_READWRITE not PAGE_EXECUTE_READWRITE
coleenp
parents: 1615
diff changeset
  3256
// checking.
823
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3257
bool os::protect_memory(char* addr, size_t bytes, ProtType prot,
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3258
                        bool is_committed) {
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3259
  unsigned int p = 0;
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3260
  switch (prot) {
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3261
  case MEM_PROT_NONE: p = PROT_NONE; break;
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3262
  case MEM_PROT_READ: p = PROT_READ; break;
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3263
  case MEM_PROT_RW:   p = PROT_READ|PROT_WRITE; break;
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3264
  case MEM_PROT_RWX:  p = PROT_READ|PROT_WRITE|PROT_EXEC; break;
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3265
  default:
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3266
    ShouldNotReachHere();
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3267
  }
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3268
  // is_committed is unused.
9a5271881bc0 6716785: implicit null checks not triggering with CompressedOops
coleenp
parents: 670
diff changeset
  3269
  return solaris_mprotect(addr, bytes, p);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3270
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3271
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3272
// guard_memory and unguard_memory only happens within stack guard pages.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3273
// Since ISM pertains only to the heap, guard and unguard memory should not
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3274
/// happen with an ISM region.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3275
bool os::guard_memory(char* addr, size_t bytes) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3276
  return solaris_mprotect(addr, bytes, PROT_NONE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3277
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3278
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3279
bool os::unguard_memory(char* addr, size_t bytes) {
1664
fc9ed50498fb 6727377: VM stack guard pages on Windows should PAGE_READWRITE not PAGE_EXECUTE_READWRITE
coleenp
parents: 1615
diff changeset
  3280
  return solaris_mprotect(addr, bytes, PROT_READ|PROT_WRITE);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3281
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3282
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3283
// Large page support
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3284
static size_t _large_page_size = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3285
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3286
// Insertion sort for small arrays (descending order).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3287
static void insertion_sort_descending(size_t* array, int len) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3288
  for (int i = 0; i < len; i++) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3289
    size_t val = array[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3290
    for (size_t key = i; key > 0 && array[key - 1] < val; --key) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3291
      size_t tmp = array[key];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3292
      array[key] = array[key - 1];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3293
      array[key - 1] = tmp;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3294
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3295
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3296
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3297
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3298
bool os::Solaris::mpss_sanity_check(bool warn, size_t* page_size) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3299
  const unsigned int usable_count = VM_Version::page_size_count();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3300
  if (usable_count == 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3301
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3302
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3303
10272
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3304
  // Find the right getpagesizes interface.  When solaris 11 is the minimum
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3305
  // build platform, getpagesizes() (without the '2') can be called directly.
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3306
  typedef int (*gps_t)(size_t[], int);
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3307
  gps_t gps_func = CAST_TO_FN_PTR(gps_t, dlsym(RTLD_DEFAULT, "getpagesizes2"));
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3308
  if (gps_func == NULL) {
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3309
    gps_func = CAST_TO_FN_PTR(gps_t, dlsym(RTLD_DEFAULT, "getpagesizes"));
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3310
    if (gps_func == NULL) {
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3311
      if (warn) {
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3312
        warning("MPSS is not supported by the operating system.");
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3313
      }
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3314
      return false;
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3315
    }
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3316
  }
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3317
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3318
  // Fill the array of page sizes.
10272
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3319
  int n = (*gps_func)(_page_sizes, page_sizes_max);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3320
  assert(n > 0, "Solaris bug?");
10272
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3321
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3322
  if (n == page_sizes_max) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3323
    // Add a sentinel value (necessary only if the array was completely filled
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3324
    // since it is static (zeroed at initialization)).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3325
    _page_sizes[--n] = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3326
    DEBUG_ONLY(warning("increase the size of the os::_page_sizes array.");)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3327
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3328
  assert(_page_sizes[n] == 0, "missing sentinel");
10272
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3329
  trace_page_sizes("available page sizes", _page_sizes, n);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3330
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3331
  if (n == 1) return false;     // Only one page size available.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3332
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3333
  // Skip sizes larger than 4M (or LargePageSizeInBytes if it was set) and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3334
  // select up to usable_count elements.  First sort the array, find the first
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3335
  // acceptable value, then copy the usable sizes to the top of the array and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3336
  // trim the rest.  Make sure to include the default page size :-).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3337
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3338
  // A better policy could get rid of the 4M limit by taking the sizes of the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3339
  // important VM memory regions (java heap and possibly the code cache) into
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3340
  // account.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3341
  insertion_sort_descending(_page_sizes, n);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3342
  const size_t size_limit =
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3343
    FLAG_IS_DEFAULT(LargePageSizeInBytes) ? 4 * M : LargePageSizeInBytes;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3344
  int beg;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3345
  for (beg = 0; beg < n && _page_sizes[beg] > size_limit; ++beg) /* empty */ ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3346
  const int end = MIN2((int)usable_count, n) - 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3347
  for (int cur = 0; cur < end; ++cur, ++beg) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3348
    _page_sizes[cur] = _page_sizes[beg];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3349
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3350
  _page_sizes[end] = vm_page_size();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3351
  _page_sizes[end + 1] = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3352
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3353
  if (_page_sizes[end] > _page_sizes[end - 1]) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3354
    // Default page size is not the smallest; sort again.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3355
    insertion_sort_descending(_page_sizes, end + 1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3356
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3357
  *page_size = _page_sizes[0];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3358
10272
9377a9510c83 6791672: enable 1G and larger pages on solaris
jcoomes
parents: 10023
diff changeset
  3359
  trace_page_sizes("usable page sizes", _page_sizes, end + 1);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3360
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3361
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3362
9419
f0360dfe734d 7040485: Use transparent huge page on linux by default
iveresov
parents: 9341
diff changeset
  3363
void os::large_page_init() {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3364
  if (UseLargePages) {
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3365
    // print a warning if any large page related flag is specified on command line
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3366
    bool warn_on_failure = !FLAG_IS_DEFAULT(UseLargePages)        ||
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3367
                           !FLAG_IS_DEFAULT(LargePageSizeInBytes);
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3368
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3369
    UseLargePages = Solaris::mpss_sanity_check(warn_on_failure, &_large_page_size);
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3370
  }
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3371
}
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3372
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3373
bool os::Solaris::setup_large_pages(caddr_t start, size_t bytes, size_t align) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3374
  // Signal to OS that we want large pages for addresses
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3375
  // from addr, addr + bytes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3376
  struct memcntl_mha mpss_struct;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3377
  mpss_struct.mha_cmd = MHA_MAPSIZE_VA;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3378
  mpss_struct.mha_pagesize = align;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3379
  mpss_struct.mha_flags = 0;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3380
  // Upon successful completion, memcntl() returns 0
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3381
  if (memcntl(start, bytes, MC_HAT_ADVISE, (caddr_t) &mpss_struct, 0, 0)) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3382
    debug_only(warning("Attempt to use MPSS failed."));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3383
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3384
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3385
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3386
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3387
19546
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  3388
char* os::reserve_memory_special(size_t size, size_t alignment, char* addr, bool exec) {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3389
  fatal("os::reserve_memory_special should not be called on Solaris.");
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3390
  return NULL;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3391
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3392
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3393
bool os::release_memory_special(char* base, size_t bytes) {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3394
  fatal("os::release_memory_special should not be called on Solaris.");
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3395
  return false;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3396
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3397
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3398
size_t os::large_page_size() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3399
  return _large_page_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3400
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3401
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3402
// MPSS allows application to commit large page memory on demand; with ISM
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3403
// the entire memory region must be allocated as shared memory.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3404
bool os::can_commit_large_page_memory() {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3405
  return true;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3406
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3407
252
050143a0dbfb 6642862: Code cache allocation fails with large pages after 6588638
jcoomes
parents: 234
diff changeset
  3408
bool os::can_execute_large_page_memory() {
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3409
  return true;
252
050143a0dbfb 6642862: Code cache allocation fails with large pages after 6588638
jcoomes
parents: 234
diff changeset
  3410
}
050143a0dbfb 6642862: Code cache allocation fails with large pages after 6588638
jcoomes
parents: 234
diff changeset
  3411
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3412
static int os_sleep(jlong millis, bool interruptible) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3413
  const jlong limit = INT_MAX;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3414
  jlong prevtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3415
  int res;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3416
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3417
  while (millis > limit) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3418
    if ((res = os_sleep(limit, interruptible)) != OS_OK)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3419
      return res;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3420
    millis -= limit;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3421
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3422
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3423
  // Restart interrupted polls with new parameters until the proper delay
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3424
  // has been completed.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3425
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3426
  prevtime = getTimeMillis();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3427
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3428
  while (millis > 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3429
    jlong newtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3430
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3431
    if (!interruptible) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3432
      // Following assert fails for os::yield_all:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3433
      // assert(!thread->is_Java_thread(), "must not be java thread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3434
      res = poll(NULL, 0, millis);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3435
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3436
      JavaThread *jt = JavaThread::current();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3437
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3438
      INTERRUPTIBLE_NORESTART_VM_ALWAYS(poll(NULL, 0, millis), res, jt,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3439
        os::Solaris::clear_interrupted);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3440
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3441
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3442
    // INTERRUPTIBLE_NORESTART_VM_ALWAYS returns res == OS_INTRPT for
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3443
    // thread.Interrupt.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3444
7392
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3445
    // See c/r 6751923. Poll can return 0 before time
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3446
    // has elapsed if time is set via clock_settime (as NTP does).
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3447
    // res == 0 if poll timed out (see man poll RETURN VALUES)
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3448
    // using the logic below checks that we really did
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3449
    // sleep at least "millis" if not we'll sleep again.
27133691a020 6751923: JNDI wake up when clock_settime() is called
zgu
parents: 6964
diff changeset
  3450
    if( ( res == 0 ) || ((res == OS_ERR) && (errno == EINTR))) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3451
      newtime = getTimeMillis();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3452
      assert(newtime >= prevtime, "time moving backwards");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3453
    /* Doing prevtime and newtime in microseconds doesn't help precision,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3454
       and trying to round up to avoid lost milliseconds can result in a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3455
       too-short delay. */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3456
      millis -= newtime - prevtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3457
      if(millis <= 0)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3458
        return OS_OK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3459
      prevtime = newtime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3460
    } else
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3461
      return res;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3462
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3463
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3464
  return OS_OK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3465
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3466
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3467
// Read calls from inside the vm need to perform state transitions
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3468
size_t os::read(int fd, void *buf, unsigned int nBytes) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3469
  INTERRUPTIBLE_RETURN_INT_VM(::read(fd, buf, nBytes), os::Solaris::clear_interrupted);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3470
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3471
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  3472
size_t os::restartable_read(int fd, void *buf, unsigned int nBytes) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  3473
  INTERRUPTIBLE_RETURN_INT(::read(fd, buf, nBytes), os::Solaris::clear_interrupted);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  3474
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  3475
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3476
int os::sleep(Thread* thread, jlong millis, bool interruptible) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3477
  assert(thread == Thread::current(),  "thread consistency check");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3478
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3479
  // TODO-FIXME: this should be removed.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3480
  // On Solaris machines (especially 2.5.1) we found that sometimes the VM gets into a live lock
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3481
  // situation with a JavaThread being starved out of a lwp. The kernel doesn't seem to generate
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3482
  // a SIGWAITING signal which would enable the threads library to create a new lwp for the starving
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3483
  // thread. We suspect that because the Watcher thread keeps waking up at periodic intervals the kernel
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3484
  // is fooled into believing that the system is making progress. In the code below we block the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3485
  // the watcher thread while safepoint is in progress so that it would not appear as though the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3486
  // system is making progress.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3487
  if (!Solaris::T2_libthread() &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3488
      thread->is_Watcher_thread() && SafepointSynchronize::is_synchronizing() && !Arguments::has_profile()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3489
    // We now try to acquire the threads lock. Since this lock is held by the VM thread during
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3490
    // the entire safepoint, the watcher thread will  line up here during the safepoint.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3491
    Threads_lock->lock_without_safepoint_check();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3492
    Threads_lock->unlock();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3493
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3494
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3495
  if (thread->is_Java_thread()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3496
    // This is a JavaThread so we honor the _thread_blocked protocol
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3497
    // even for sleeps of 0 milliseconds. This was originally done
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3498
    // as a workaround for bug 4338139. However, now we also do it
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3499
    // to honor the suspend-equivalent protocol.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3500
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3501
    JavaThread *jt = (JavaThread *) thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3502
    ThreadBlockInVM tbivm(jt);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3503
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3504
    jt->set_suspend_equivalent();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3505
    // cleared by handle_special_suspend_equivalent_condition() or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3506
    // java_suspend_self() via check_and_wait_while_suspended()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3507
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3508
    int ret_code;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3509
    if (millis <= 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3510
      thr_yield();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3511
      ret_code = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3512
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3513
      // The original sleep() implementation did not create an
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3514
      // OSThreadWaitState helper for sleeps of 0 milliseconds.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3515
      // I'm preserving that decision for now.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3516
      OSThreadWaitState osts(jt->osthread(), false /* not Object.wait() */);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3517
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3518
      ret_code = os_sleep(millis, interruptible);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3519
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3520
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3521
    // were we externally suspended while we were waiting?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3522
    jt->check_and_wait_while_suspended();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3523
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3524
    return ret_code;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3525
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3526
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3527
  // non-JavaThread from this point on:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3528
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3529
  if (millis <= 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3530
    thr_yield();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3531
    return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3532
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3533
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3534
  OSThreadWaitState osts(thread->osthread(), false /* not Object.wait() */);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3535
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3536
  return os_sleep(millis, interruptible);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3537
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3538
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3539
int os::naked_sleep() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3540
  // %% make the sleep time an integer flag. for now use 1 millisec.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3541
  return os_sleep(1, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3542
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3543
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3544
// Sleep forever; naked call to OS-specific sleep; use with CAUTION
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3545
void os::infinite_sleep() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3546
  while (true) {    // sleep forever ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3547
    ::sleep(100);   // ... 100 seconds at a time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3548
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3549
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3550
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3551
// Used to convert frequent JVM_Yield() to nops
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3552
bool os::dont_yield() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3553
  if (DontYieldALot) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3554
    static hrtime_t last_time = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3555
    hrtime_t diff = getTimeNanos() - last_time;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3556
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3557
    if (diff < DontYieldALotInterval * 1000000)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3558
      return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3559
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3560
    last_time += diff;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3561
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3562
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3563
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3564
  else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3565
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3566
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3567
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3568
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3569
// Caveat: Solaris os::yield() causes a thread-state transition whereas
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3570
// the linux and win32 implementations do not.  This should be checked.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3571
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3572
void os::yield() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3573
  // Yields to all threads with same or greater priority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3574
  os::sleep(Thread::current(), 0, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3575
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3576
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3577
// Note that yield semantics are defined by the scheduling class to which
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3578
// the thread currently belongs.  Typically, yield will _not yield to
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3579
// other equal or higher priority threads that reside on the dispatch queues
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3580
// of other CPUs.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3581
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3582
os::YieldResult os::NakedYield() { thr_yield(); return os::YIELD_UNKNOWN; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3583
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3584
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3585
// On Solaris we found that yield_all doesn't always yield to all other threads.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3586
// There have been cases where there is a thread ready to execute but it doesn't
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3587
// get an lwp as the VM thread continues to spin with sleeps of 1 millisecond.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3588
// The 1 millisecond wait doesn't seem long enough for the kernel to issue a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3589
// SIGWAITING signal which will cause a new lwp to be created. So we count the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3590
// number of times yield_all is called in the one loop and increase the sleep
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3591
// time after 8 attempts. If this fails too we increase the concurrency level
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3592
// so that the starving thread would get an lwp
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3593
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3594
void os::yield_all(int attempts) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3595
  // Yields to all threads, including threads with lower priorities
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3596
  if (attempts == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3597
    os::sleep(Thread::current(), 1, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3598
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3599
    int iterations = attempts % 30;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3600
    if (iterations == 0 && !os::Solaris::T2_libthread()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3601
      // thr_setconcurrency and _getconcurrency make sense only under T1.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3602
      int noofLWPS = thr_getconcurrency();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3603
      if (noofLWPS < (Threads::number_of_threads() + 2)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3604
        thr_setconcurrency(thr_getconcurrency() + 1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3605
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3606
    } else if (iterations < 25) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3607
      os::sleep(Thread::current(), 1, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3608
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3609
      os::sleep(Thread::current(), 10, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3610
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3611
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3612
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3613
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3614
// Called from the tight loops to possibly influence time-sharing heuristics
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3615
void os::loop_breaker(int attempts) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3616
  os::yield_all(attempts);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3617
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3618
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3619
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3620
// Interface for setting lwp priorities.  If we are using T2 libthread,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3621
// which forces the use of BoundThreads or we manually set UseBoundThreads,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3622
// all of our threads will be assigned to real lwp's.  Using the thr_setprio
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3623
// function is meaningless in this mode so we must adjust the real lwp's priority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3624
// The routines below implement the getting and setting of lwp priorities.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3625
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3626
// Note: There are three priority scales used on Solaris.  Java priotities
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3627
//       which range from 1 to 10, libthread "thr_setprio" scale which range
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3628
//       from 0 to 127, and the current scheduling class of the process we
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3629
//       are running in.  This is typically from -60 to +60.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3630
//       The setting of the lwp priorities in done after a call to thr_setprio
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3631
//       so Java priorities are mapped to libthread priorities and we map from
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3632
//       the latter to lwp priorities.  We don't keep priorities stored in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3633
//       Java priorities since some of our worker threads want to set priorities
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3634
//       higher than all Java threads.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3635
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3636
// For related information:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3637
// (1)  man -s 2 priocntl
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3638
// (2)  man -s 4 priocntl
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3639
// (3)  man dispadmin
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3640
// =    librt.so
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3641
// =    libthread/common/rtsched.c - thrp_setlwpprio().
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3642
// =    ps -cL <pid> ... to validate priority.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3643
// =    sched_get_priority_min and _max
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3644
//              pthread_create
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3645
//              sched_setparam
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3646
//              pthread_setschedparam
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3647
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3648
// Assumptions:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3649
// +    We assume that all threads in the process belong to the same
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3650
//              scheduling class.   IE. an homogenous process.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3651
// +    Must be root or in IA group to change change "interactive" attribute.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3652
//              Priocntl() will fail silently.  The only indication of failure is when
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3653
//              we read-back the value and notice that it hasn't changed.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3654
// +    Interactive threads enter the runq at the head, non-interactive at the tail.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3655
// +    For RT, change timeslice as well.  Invariant:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3656
//              constant "priority integral"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3657
//              Konst == TimeSlice * (60-Priority)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3658
//              Given a priority, compute appropriate timeslice.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3659
// +    Higher numerical values have higher priority.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3660
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3661
// sched class attributes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3662
typedef struct {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3663
        int   schedPolicy;              // classID
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3664
        int   maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3665
        int   minPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3666
} SchedInfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3667
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3668
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3669
static SchedInfo tsLimits, iaLimits, rtLimits, fxLimits;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3670
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3671
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3672
static int  ReadBackValidate = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3673
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3674
static int  myClass     = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3675
static int  myMin       = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3676
static int  myMax       = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3677
static int  myCur       = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3678
static bool priocntl_enable = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3679
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3680
static const int criticalPrio = 60; // FX/60 is critical thread class/priority on T4
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3681
static int java_MaxPriority_to_os_priority = 0; // Saved mapping
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3682
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3683
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3684
// lwp_priocntl_init
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3685
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3686
// Try to determine the priority scale for our process.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3687
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3688
// Return errno or 0 if OK.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3689
//
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3690
static int lwp_priocntl_init () {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3691
  int rslt;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3692
  pcinfo_t ClassInfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3693
  pcparms_t ParmInfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3694
  int i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3695
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3696
  if (!UseThreadPriorities) return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3697
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3698
  // We are using Bound threads, we need to determine our priority ranges
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3699
  if (os::Solaris::T2_libthread() || UseBoundThreads) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3700
    // If ThreadPriorityPolicy is 1, switch tables
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3701
    if (ThreadPriorityPolicy == 1) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3702
      for (i = 0 ; i < CriticalPriority+1; i++)
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3703
        os::java_to_os_priority[i] = prio_policy1[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3704
    }
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3705
    if (UseCriticalJavaThreadPriority) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3706
      // MaxPriority always maps to the FX scheduling class and criticalPrio.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3707
      // See set_native_priority() and set_lwp_class_and_priority().
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3708
      // Save original MaxPriority mapping in case attempt to
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3709
      // use critical priority fails.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3710
      java_MaxPriority_to_os_priority = os::java_to_os_priority[MaxPriority];
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3711
      // Set negative to distinguish from other priorities
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3712
      os::java_to_os_priority[MaxPriority] = -criticalPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3713
    }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3714
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3715
  // Not using Bound Threads, set to ThreadPolicy 1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3716
  else {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3717
    for ( i = 0 ; i < CriticalPriority+1; i++ ) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3718
      os::java_to_os_priority[i] = prio_policy1[i];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3719
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3720
    return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3721
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3722
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3723
  // Get IDs for a set of well-known scheduling classes.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3724
  // TODO-FIXME: GETCLINFO returns the current # of classes in the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3725
  // the system.  We should have a loop that iterates over the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3726
  // classID values, which are known to be "small" integers.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3727
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3728
  strcpy(ClassInfo.pc_clname, "TS");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3729
  ClassInfo.pc_cid = -1;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3730
  rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3731
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3732
  assert(ClassInfo.pc_cid != -1, "cid for TS class is -1");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3733
  tsLimits.schedPolicy = ClassInfo.pc_cid;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3734
  tsLimits.maxPrio = ((tsinfo_t*)ClassInfo.pc_clinfo)->ts_maxupri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3735
  tsLimits.minPrio = -tsLimits.maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3736
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3737
  strcpy(ClassInfo.pc_clname, "IA");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3738
  ClassInfo.pc_cid = -1;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3739
  rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3740
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3741
  assert(ClassInfo.pc_cid != -1, "cid for IA class is -1");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3742
  iaLimits.schedPolicy = ClassInfo.pc_cid;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3743
  iaLimits.maxPrio = ((iainfo_t*)ClassInfo.pc_clinfo)->ia_maxupri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3744
  iaLimits.minPrio = -iaLimits.maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3745
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3746
  strcpy(ClassInfo.pc_clname, "RT");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3747
  ClassInfo.pc_cid = -1;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3748
  rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3749
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3750
  assert(ClassInfo.pc_cid != -1, "cid for RT class is -1");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3751
  rtLimits.schedPolicy = ClassInfo.pc_cid;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3752
  rtLimits.maxPrio = ((rtinfo_t*)ClassInfo.pc_clinfo)->rt_maxpri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3753
  rtLimits.minPrio = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3754
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3755
  strcpy(ClassInfo.pc_clname, "FX");
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3756
  ClassInfo.pc_cid = -1;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3757
  rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3758
  if (rslt < 0) return errno;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3759
  assert(ClassInfo.pc_cid != -1, "cid for FX class is -1");
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3760
  fxLimits.schedPolicy = ClassInfo.pc_cid;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3761
  fxLimits.maxPrio = ((fxinfo_t*)ClassInfo.pc_clinfo)->fx_maxupri;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3762
  fxLimits.minPrio = 0;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3763
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3764
  // Query our "current" scheduling class.
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3765
  // This will normally be IA, TS or, rarely, FX or RT.
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3766
  memset(&ParmInfo, 0, sizeof(ParmInfo));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3767
  ParmInfo.pc_cid = PC_CLNULL;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3768
  rslt = priocntl(P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3769
  if (rslt < 0) return errno;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3770
  myClass = ParmInfo.pc_cid;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3771
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3772
  // We now know our scheduling classId, get specific information
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3773
  // about the class.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3774
  ClassInfo.pc_cid = myClass;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3775
  ClassInfo.pc_clname[0] = 0;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3776
  rslt = priocntl((idtype)0, 0, PC_GETCLINFO, (caddr_t)&ClassInfo);
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3777
  if (rslt < 0) return errno;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3778
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3779
  if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3780
    tty->print_cr("lwp_priocntl_init: Class=%d(%s)...", myClass, ClassInfo.pc_clname);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3781
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3782
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3783
  memset(&ParmInfo, 0, sizeof(pcparms_t));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3784
  ParmInfo.pc_cid = PC_CLNULL;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3785
  rslt = priocntl(P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3786
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3787
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3788
  if (ParmInfo.pc_cid == rtLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3789
    myMin = rtLimits.minPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3790
    myMax = rtLimits.maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3791
  } else if (ParmInfo.pc_cid == iaLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3792
    iaparms_t *iaInfo  = (iaparms_t*)ParmInfo.pc_clparms;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3793
    myMin = iaLimits.minPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3794
    myMax = iaLimits.maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3795
    myMax = MIN2(myMax, (int)iaInfo->ia_uprilim);       // clamp - restrict
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3796
  } else if (ParmInfo.pc_cid == tsLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3797
    tsparms_t *tsInfo  = (tsparms_t*)ParmInfo.pc_clparms;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3798
    myMin = tsLimits.minPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3799
    myMax = tsLimits.maxPrio;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3800
    myMax = MIN2(myMax, (int)tsInfo->ts_uprilim);       // clamp - restrict
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3801
  } else if (ParmInfo.pc_cid == fxLimits.schedPolicy) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3802
    fxparms_t *fxInfo = (fxparms_t*)ParmInfo.pc_clparms;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3803
    myMin = fxLimits.minPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3804
    myMax = fxLimits.maxPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3805
    myMax = MIN2(myMax, (int)fxInfo->fx_uprilim);       // clamp - restrict
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3806
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3807
    // No clue - punt
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3808
    if (ThreadPriorityVerbose)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3809
      tty->print_cr ("Unknown scheduling class: %s ... \n", ClassInfo.pc_clname);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3810
    return EINVAL;      // no clue, punt
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3811
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3812
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3813
  if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3814
    tty->print_cr ("Thread priority Range: [%d..%d]\n", myMin, myMax);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3815
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3816
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3817
  priocntl_enable = true;  // Enable changing priorities
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3818
  return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3819
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3820
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3821
#define IAPRI(x)        ((iaparms_t *)((x).pc_clparms))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3822
#define RTPRI(x)        ((rtparms_t *)((x).pc_clparms))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3823
#define TSPRI(x)        ((tsparms_t *)((x).pc_clparms))
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3824
#define FXPRI(x)        ((fxparms_t *)((x).pc_clparms))
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3825
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3826
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3827
// scale_to_lwp_priority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3828
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3829
// Convert from the libthread "thr_setprio" scale to our current
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3830
// lwp scheduling class scale.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3831
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3832
static
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3833
int     scale_to_lwp_priority (int rMin, int rMax, int x)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3834
{
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3835
  int v;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3836
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3837
  if (x == 127) return rMax;            // avoid round-down
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3838
    v = (((x*(rMax-rMin)))/128)+rMin;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3839
  return v;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3840
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3841
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3842
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3843
// set_lwp_class_and_priority
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3844
//
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3845
// Set the class and priority of the lwp.  This call should only
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3846
// be made when using bound threads (T2 threads are bound by default).
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3847
//
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3848
int set_lwp_class_and_priority(int ThreadID, int lwpid,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3849
                               int newPrio, int new_class, bool scale) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3850
  int rslt;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3851
  int Actual, Expected, prv;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3852
  pcparms_t ParmInfo;                   // for GET-SET
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3853
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3854
  pcparms_t ReadBack;                   // for readback
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3855
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3856
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3857
  // Set priority via PC_GETPARMS, update, PC_SETPARMS
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3858
  // Query current values.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3859
  // TODO: accelerate this by eliminating the PC_GETPARMS call.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3860
  // Cache "pcparms_t" in global ParmCache.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3861
  // TODO: elide set-to-same-value
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3862
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3863
  // If something went wrong on init, don't change priorities.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3864
  if ( !priocntl_enable ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3865
    if (ThreadPriorityVerbose)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3866
      tty->print_cr("Trying to set priority but init failed, ignoring");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3867
    return EINVAL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3868
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3869
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3870
  // If lwp hasn't started yet, just return
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3871
  // the _start routine will call us again.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3872
  if ( lwpid <= 0 ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3873
    if (ThreadPriorityVerbose) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3874
      tty->print_cr ("deferring the set_lwp_class_and_priority of thread "
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3875
                     INTPTR_FORMAT " to %d, lwpid not set",
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3876
                     ThreadID, newPrio);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3877
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3878
    return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3879
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3880
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3881
  if (ThreadPriorityVerbose) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3882
    tty->print_cr ("set_lwp_class_and_priority("
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3883
                   INTPTR_FORMAT "@" INTPTR_FORMAT " %d) ",
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3884
                   ThreadID, lwpid, newPrio);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3885
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3886
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3887
  memset(&ParmInfo, 0, sizeof(pcparms_t));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3888
  ParmInfo.pc_cid = PC_CLNULL;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3889
  rslt = priocntl(P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ParmInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3890
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3891
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3892
  int cur_class = ParmInfo.pc_cid;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3893
  ParmInfo.pc_cid = (id_t)new_class;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3894
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3895
  if (new_class == rtLimits.schedPolicy) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3896
    rtparms_t *rtInfo  = (rtparms_t*)ParmInfo.pc_clparms;
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3897
    rtInfo->rt_pri     = scale ? scale_to_lwp_priority(rtLimits.minPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3898
                                                       rtLimits.maxPrio, newPrio)
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3899
                               : newPrio;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3900
    rtInfo->rt_tqsecs  = RT_NOCHANGE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3901
    rtInfo->rt_tqnsecs = RT_NOCHANGE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3902
    if (ThreadPriorityVerbose) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3903
      tty->print_cr("RT: %d->%d\n", newPrio, rtInfo->rt_pri);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3904
    }
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3905
  } else if (new_class == iaLimits.schedPolicy) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3906
    iaparms_t* iaInfo  = (iaparms_t*)ParmInfo.pc_clparms;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3907
    int maxClamped     = MIN2(iaLimits.maxPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3908
                              cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3909
                                ? (int)iaInfo->ia_uprilim : iaLimits.maxPrio);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3910
    iaInfo->ia_upri    = scale ? scale_to_lwp_priority(iaLimits.minPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3911
                                                       maxClamped, newPrio)
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3912
                               : newPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3913
    iaInfo->ia_uprilim = cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3914
                           ? IA_NOCHANGE : (pri_t)iaLimits.maxPrio;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3915
    iaInfo->ia_mode    = IA_NOCHANGE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3916
    if (ThreadPriorityVerbose) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3917
      tty->print_cr("IA: [%d...%d] %d->%d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3918
                    iaLimits.minPrio, maxClamped, newPrio, iaInfo->ia_upri);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3919
    }
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3920
  } else if (new_class == tsLimits.schedPolicy) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3921
    tsparms_t* tsInfo  = (tsparms_t*)ParmInfo.pc_clparms;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3922
    int maxClamped     = MIN2(tsLimits.maxPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3923
                              cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3924
                                ? (int)tsInfo->ts_uprilim : tsLimits.maxPrio);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3925
    tsInfo->ts_upri    = scale ? scale_to_lwp_priority(tsLimits.minPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3926
                                                       maxClamped, newPrio)
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3927
                               : newPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3928
    tsInfo->ts_uprilim = cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3929
                           ? TS_NOCHANGE : (pri_t)tsLimits.maxPrio;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3930
    if (ThreadPriorityVerbose) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3931
      tty->print_cr("TS: [%d...%d] %d->%d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3932
                    tsLimits.minPrio, maxClamped, newPrio, tsInfo->ts_upri);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3933
    }
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3934
  } else if (new_class == fxLimits.schedPolicy) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3935
    fxparms_t* fxInfo  = (fxparms_t*)ParmInfo.pc_clparms;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3936
    int maxClamped     = MIN2(fxLimits.maxPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3937
                              cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3938
                                ? (int)fxInfo->fx_uprilim : fxLimits.maxPrio);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3939
    fxInfo->fx_upri    = scale ? scale_to_lwp_priority(fxLimits.minPrio,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3940
                                                       maxClamped, newPrio)
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3941
                               : newPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3942
    fxInfo->fx_uprilim = cur_class == new_class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3943
                           ? FX_NOCHANGE : (pri_t)fxLimits.maxPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3944
    fxInfo->fx_tqsecs  = FX_NOCHANGE;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3945
    fxInfo->fx_tqnsecs = FX_NOCHANGE;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3946
    if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3947
      tty->print_cr("FX: [%d...%d] %d->%d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3948
                    fxLimits.minPrio, maxClamped, newPrio, fxInfo->fx_upri);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3949
    }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3950
  } else {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3951
    if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3952
      tty->print_cr("Unknown new scheduling class %d\n", new_class);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3953
    }
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3954
    return EINVAL;    // no clue, punt
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3955
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3956
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3957
  rslt = priocntl(P_LWPID, lwpid, PC_SETPARMS, (caddr_t)&ParmInfo);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3958
  if (ThreadPriorityVerbose && rslt) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3959
    tty->print_cr ("PC_SETPARMS ->%d %d\n", rslt, errno);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3960
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3961
  if (rslt < 0) return errno;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3962
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3963
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3964
  // Sanity check: read back what we just attempted to set.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3965
  // In theory it could have changed in the interim ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3966
  //
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3967
  // The priocntl system call is tricky.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3968
  // Sometimes it'll validate the priority value argument and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3969
  // return EINVAL if unhappy.  At other times it fails silently.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3970
  // Readbacks are prudent.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3971
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3972
  if (!ReadBackValidate) return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3973
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3974
  memset(&ReadBack, 0, sizeof(pcparms_t));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3975
  ReadBack.pc_cid = PC_CLNULL;
18702
e6fa43c6d215 8016749: -XX:+UseISM fails an assert(obj->is_oop()) when running SPECjbb2005
anoll
parents: 18086
diff changeset
  3976
  rslt = priocntl(P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ReadBack);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3977
  assert(rslt >= 0, "priocntl failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3978
  Actual = Expected = 0xBAD;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3979
  assert(ParmInfo.pc_cid == ReadBack.pc_cid, "cid's don't match");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3980
  if (ParmInfo.pc_cid == rtLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3981
    Actual   = RTPRI(ReadBack)->rt_pri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3982
    Expected = RTPRI(ParmInfo)->rt_pri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3983
  } else if (ParmInfo.pc_cid == iaLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3984
    Actual   = IAPRI(ReadBack)->ia_upri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3985
    Expected = IAPRI(ParmInfo)->ia_upri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3986
  } else if (ParmInfo.pc_cid == tsLimits.schedPolicy) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3987
    Actual   = TSPRI(ReadBack)->ts_upri;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3988
    Expected = TSPRI(ParmInfo)->ts_upri;
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3989
  } else if (ParmInfo.pc_cid == fxLimits.schedPolicy) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3990
    Actual   = FXPRI(ReadBack)->fx_upri;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3991
    Expected = FXPRI(ParmInfo)->fx_upri;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3992
  } else {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3993
    if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3994
      tty->print_cr("set_lwp_class_and_priority: unexpected class in readback: %d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  3995
                    ParmInfo.pc_cid);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3996
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3997
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3998
489c9b5090e2 Initial load
duke
parents:
diff changeset
  3999
  if (Actual != Expected) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4000
    if (ThreadPriorityVerbose) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4001
      tty->print_cr ("set_lwp_class_and_priority(%d %d) Class=%d: actual=%d vs expected=%d\n",
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4002
                     lwpid, newPrio, ReadBack.pc_cid, Actual, Expected);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4003
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4004
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4005
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4006
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4007
  return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4008
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4009
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4010
// Solaris only gives access to 128 real priorities at a time,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4011
// so we expand Java's ten to fill this range.  This would be better
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4012
// if we dynamically adjusted relative priorities.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4013
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4014
// The ThreadPriorityPolicy option allows us to select 2 different
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4015
// priority scales.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4016
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4017
// ThreadPriorityPolicy=0
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4018
// Since the Solaris' default priority is MaximumPriority, we do not
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4019
// set a priority lower than Max unless a priority lower than
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4020
// NormPriority is requested.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4021
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4022
// ThreadPriorityPolicy=1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4023
// This mode causes the priority table to get filled with
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4024
// linear values.  NormPriority get's mapped to 50% of the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4025
// Maximum priority an so on.  This will cause VM threads
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4026
// to get unfair treatment against other Solaris processes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4027
// which do not explicitly alter their thread priorities.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4028
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4029
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4030
int os::java_to_os_priority[CriticalPriority + 1] = {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4031
  -99999,         // 0 Entry should never be used
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4032
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4033
  0,              // 1 MinPriority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4034
  32,             // 2
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4035
  64,             // 3
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4036
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4037
  96,             // 4
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4038
  127,            // 5 NormPriority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4039
  127,            // 6
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4040
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4041
  127,            // 7
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4042
  127,            // 8
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4043
  127,            // 9 NearMaxPriority
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4044
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4045
  127,            // 10 MaxPriority
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4046
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4047
  -criticalPrio   // 11 CriticalPriority
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4048
};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4049
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4050
OSReturn os::set_native_priority(Thread* thread, int newpri) {
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4051
  OSThread* osthread = thread->osthread();
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4052
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4053
  // Save requested priority in case the thread hasn't been started
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4054
  osthread->set_native_priority(newpri);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4055
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4056
  // Check for critical priority request
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4057
  bool fxcritical = false;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4058
  if (newpri == -criticalPrio) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4059
    fxcritical = true;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4060
    newpri = criticalPrio;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4061
  }
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4062
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4063
  assert(newpri >= MinimumPriority && newpri <= MaximumPriority, "bad priority mapping");
11601
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4064
  if (!UseThreadPriorities) return OS_OK;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4065
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4066
  int status = 0;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4067
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4068
  if (!fxcritical) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4069
    // Use thr_setprio only if we have a priority that thr_setprio understands
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4070
    status = thr_setprio(thread->osthread()->thread_id(), newpri);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4071
  }
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4072
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4073
  if (os::Solaris::T2_libthread() ||
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4074
      (UseBoundThreads && osthread->is_vm_created())) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4075
    int lwp_status =
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4076
      set_lwp_class_and_priority(osthread->thread_id(),
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4077
                                 osthread->lwp_id(),
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4078
                                 newpri,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4079
                                 fxcritical ? fxLimits.schedPolicy : myClass,
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4080
                                 !fxcritical);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4081
    if (lwp_status != 0 && fxcritical) {
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4082
      // Try again, this time without changing the scheduling class
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4083
      newpri = java_MaxPriority_to_os_priority;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4084
      lwp_status = set_lwp_class_and_priority(osthread->thread_id(),
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4085
                                              osthread->lwp_id(),
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4086
                                              newpri, myClass, false);
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4087
    }
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4088
    status |= lwp_status;
f359304c1856 7082553: Interpret Thread.setPriority(Thread.MAX_PRIORITY) to mean FX60 on Solaris 10 and 11
phh
parents: 11483
diff changeset
  4089
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4090
  return (status == 0) ? OS_OK : OS_ERR;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4091
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4092
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4093
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4094
OSReturn os::get_native_priority(const Thread* const thread, int *priority_ptr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4095
  int p;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4096
  if ( !UseThreadPriorities ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4097
    *priority_ptr = NormalPriority;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4098
    return OS_OK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4099
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4100
  int status = thr_getprio(thread->osthread()->thread_id(), &p);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4101
  if (status != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4102
    return OS_ERR;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4103
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4104
  *priority_ptr = p;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4105
  return OS_OK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4106
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4107
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4108
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4109
// Hint to the underlying OS that a task switch would not be good.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4110
// Void return because it's a hint and can fail.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4111
void os::hint_no_preempt() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4112
  schedctl_start(schedctl_init());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4113
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4114
18025
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4115
static void resume_clear_context(OSThread *osthread) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4116
  osthread->set_ucontext(NULL);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4117
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4118
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4119
static void suspend_save_context(OSThread *osthread, ucontext_t* context) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4120
  osthread->set_ucontext(context);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4121
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4122
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4123
static Semaphore sr_semaphore;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4124
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4125
void os::Solaris::SR_handler(Thread* thread, ucontext_t* uc) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4126
  // Save and restore errno to avoid confusing native code with EINTR
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4127
  // after sigsuspend.
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4128
  int old_errno = errno;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4129
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4130
  OSThread* osthread = thread->osthread();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4131
  assert(thread->is_VM_thread() || thread->is_Java_thread(), "Must be VMThread or JavaThread");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4132
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4133
  os::SuspendResume::State current = osthread->sr.state();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4134
  if (current == os::SuspendResume::SR_SUSPEND_REQUEST) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4135
    suspend_save_context(osthread, uc);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4136
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4137
    // attempt to switch the state, we assume we had a SUSPEND_REQUEST
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4138
    os::SuspendResume::State state = osthread->sr.suspended();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4139
    if (state == os::SuspendResume::SR_SUSPENDED) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4140
      sigset_t suspend_set;  // signals for sigsuspend()
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4141
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4142
      // get current set of blocked signals and unblock resume signal
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4143
      thr_sigsetmask(SIG_BLOCK, NULL, &suspend_set);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4144
      sigdelset(&suspend_set, os::Solaris::SIGasync());
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4145
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4146
      sr_semaphore.signal();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4147
      // wait here until we are resumed
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4148
      while (1) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4149
        sigsuspend(&suspend_set);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4150
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4151
        os::SuspendResume::State result = osthread->sr.running();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4152
        if (result == os::SuspendResume::SR_RUNNING) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4153
          sr_semaphore.signal();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4154
          break;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4155
        }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4156
      }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4157
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4158
    } else if (state == os::SuspendResume::SR_RUNNING) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4159
      // request was cancelled, continue
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4160
    } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4161
      ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4162
    }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4163
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4164
    resume_clear_context(osthread);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4165
  } else if (current == os::SuspendResume::SR_RUNNING) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4166
    // request was cancelled, continue
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4167
  } else if (current == os::SuspendResume::SR_WAKEUP_REQUEST) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4168
    // ignore
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4169
  } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4170
    // ignore
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4171
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4172
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4173
  errno = old_errno;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4174
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4175
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4176
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4177
void os::interrupt(Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4178
  assert(Thread::current() == thread || Threads_lock->owned_by_self(), "possibility of dangling Thread pointer");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4179
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4180
  OSThread* osthread = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4181
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4182
  int isInterrupted = osthread->interrupted();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4183
  if (!isInterrupted) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4184
      osthread->set_interrupted(true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4185
      OrderAccess::fence();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4186
      // os::sleep() is implemented with either poll (NULL,0,timeout) or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4187
      // by parking on _SleepEvent.  If the former, thr_kill will unwedge
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4188
      // the sleeper by SIGINTR, otherwise the unpark() will wake the sleeper.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4189
      ParkEvent * const slp = thread->_SleepEvent ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4190
      if (slp != NULL) slp->unpark() ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4191
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4192
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4193
  // For JSR166:  unpark after setting status but before thr_kill -dl
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4194
  if (thread->is_Java_thread()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4195
    ((JavaThread*)thread)->parker()->unpark();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4196
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4197
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4198
  // Handle interruptible wait() ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4199
  ParkEvent * const ev = thread->_ParkEvent ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4200
  if (ev != NULL) ev->unpark() ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4201
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4202
  // When events are used everywhere for os::sleep, then this thr_kill
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4203
  // will only be needed if UseVMInterruptibleIO is true.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4204
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4205
  if (!isInterrupted) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4206
    int status = thr_kill(osthread->thread_id(), os::Solaris::SIGinterrupt());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4207
    assert_status(status == 0, status, "thr_kill");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4208
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4209
    // Bump thread interruption counter
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4210
    RuntimeService::record_thread_interrupt_signaled_count();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4211
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4212
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4213
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4214
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4215
bool os::is_interrupted(Thread* thread, bool clear_interrupted) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4216
  assert(Thread::current() == thread || Threads_lock->owned_by_self(), "possibility of dangling Thread pointer");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4217
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4218
  OSThread* osthread = thread->osthread();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4219
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4220
  bool res = osthread->interrupted();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4221
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4222
  // NOTE that since there is no "lock" around these two operations,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4223
  // there is the possibility that the interrupted flag will be
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4224
  // "false" but that the interrupt event will be set. This is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4225
  // intentional. The effect of this is that Object.wait() will appear
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4226
  // to have a spurious wakeup, which is not harmful, and the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4227
  // possibility is so rare that it is not worth the added complexity
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4228
  // to add yet another lock. It has also been recommended not to put
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4229
  // the interrupted flag into the os::Solaris::Event structure,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4230
  // because it hides the issue.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4231
  if (res && clear_interrupted) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4232
    osthread->set_interrupted(false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4233
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4234
  return res;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4235
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4236
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4237
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4238
void os::print_statistics() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4239
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4240
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4241
int os::message_box(const char* title, const char* message) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4242
  int i;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4243
  fdStream err(defaultStream::error_fd());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4244
  for (i = 0; i < 78; i++) err.print_raw("=");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4245
  err.cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4246
  err.print_raw_cr(title);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4247
  for (i = 0; i < 78; i++) err.print_raw("-");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4248
  err.cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4249
  err.print_raw_cr(message);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4250
  for (i = 0; i < 78; i++) err.print_raw("=");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4251
  err.cr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4252
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4253
  char buf[16];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4254
  // Prevent process from exiting upon "read error" without consuming all CPU
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4255
  while (::read(0, buf, sizeof(buf)) <= 0) { ::sleep(100); }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4256
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4257
  return buf[0] == 'y' || buf[0] == 'Y';
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4258
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4259
18025
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4260
static int sr_notify(OSThread* osthread) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4261
  int status = thr_kill(osthread->thread_id(), os::Solaris::SIGasync());
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4262
  assert_status(status == 0, status, "thr_kill");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4263
  return status;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4264
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4265
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4266
// "Randomly" selected value for how long we want to spin
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4267
// before bailing out on suspending a thread, also how often
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4268
// we send a signal to a thread we want to resume
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4269
static const int RANDOMLY_LARGE_INTEGER = 1000000;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4270
static const int RANDOMLY_LARGE_INTEGER2 = 100;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4271
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4272
static bool do_suspend(OSThread* osthread) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4273
  assert(osthread->sr.is_running(), "thread should be running");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4274
  assert(!sr_semaphore.trywait(), "semaphore has invalid state");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4275
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4276
  // mark as suspended and send signal
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4277
  if (osthread->sr.request_suspend() != os::SuspendResume::SR_SUSPEND_REQUEST) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4278
    // failed to switch, state wasn't running?
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4279
    ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4280
    return false;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4281
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4282
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4283
  if (sr_notify(osthread) != 0) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4284
    ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4285
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4286
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4287
  // managed to send the signal and switch to SUSPEND_REQUEST, now wait for SUSPENDED
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4288
  while (true) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4289
    if (sr_semaphore.timedwait(0, 2000 * NANOSECS_PER_MILLISEC)) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4290
      break;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4291
    } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4292
      // timeout
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4293
      os::SuspendResume::State cancelled = osthread->sr.cancel_suspend();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4294
      if (cancelled == os::SuspendResume::SR_RUNNING) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4295
        return false;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4296
      } else if (cancelled == os::SuspendResume::SR_SUSPENDED) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4297
        // make sure that we consume the signal on the semaphore as well
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4298
        sr_semaphore.wait();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4299
        break;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4300
      } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4301
        ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4302
        return false;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4303
      }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4304
    }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4305
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4306
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4307
  guarantee(osthread->sr.is_suspended(), "Must be suspended");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4308
  return true;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4309
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4310
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4311
static void do_resume(OSThread* osthread) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4312
  assert(osthread->sr.is_suspended(), "thread should be suspended");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4313
  assert(!sr_semaphore.trywait(), "invalid semaphore state");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4314
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4315
  if (osthread->sr.request_wakeup() != os::SuspendResume::SR_WAKEUP_REQUEST) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4316
    // failed to switch to WAKEUP_REQUEST
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4317
    ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4318
    return;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4319
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4320
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4321
  while (true) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4322
    if (sr_notify(osthread) == 0) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4323
      if (sr_semaphore.timedwait(0, 2 * NANOSECS_PER_MILLISEC)) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4324
        if (osthread->sr.is_running()) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4325
          return;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4326
        }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4327
      }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4328
    } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4329
      ShouldNotReachHere();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4330
    }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4331
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4332
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4333
  guarantee(osthread->sr.is_running(), "Must be running!");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4334
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4335
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4336
void os::SuspendedThreadTask::internal_do_task() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4337
  if (do_suspend(_thread->osthread())) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4338
    SuspendedThreadTaskContext context(_thread, _thread->osthread()->ucontext());
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4339
    do_task(context);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4340
    do_resume(_thread->osthread());
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4341
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4342
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4343
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4344
class PcFetcher : public os::SuspendedThreadTask {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4345
public:
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4346
  PcFetcher(Thread* thread) : os::SuspendedThreadTask(thread) {}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4347
  ExtendedPC result();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4348
protected:
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4349
  void do_task(const os::SuspendedThreadTaskContext& context);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4350
private:
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4351
  ExtendedPC _epc;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4352
};
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4353
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4354
ExtendedPC PcFetcher::result() {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4355
  guarantee(is_done(), "task is not done yet.");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4356
  return _epc;
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4357
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4358
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4359
void PcFetcher::do_task(const os::SuspendedThreadTaskContext& context) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4360
  Thread* thread = context.thread();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4361
  OSThread* osthread = thread->osthread();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4362
  if (osthread->ucontext() != NULL) {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4363
    _epc = os::Solaris::ucontext_get_pc((ucontext_t *) context.ucontext());
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4364
  } else {
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4365
    // NULL context is unexpected, double-check this is the VMThread
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4366
    guarantee(thread->is_VM_thread(), "can only be called for VMThread");
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4367
  }
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4368
}
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4369
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4370
// A lightweight implementation that does not suspend the target thread and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4371
// thus returns only a hint. Used for profiling only!
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4372
ExtendedPC os::get_thread_pc(Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4373
  // Make sure that it is called by the watcher and the Threads lock is owned.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4374
  assert(Thread::current()->is_Watcher_thread(), "Must be watcher and own Threads_lock");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4375
  // For now, is only used to profile the VM Thread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4376
  assert(thread->is_VM_thread(), "Can only be called for VMThread");
18025
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4377
  PcFetcher fetcher(thread);
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4378
  fetcher.run();
b7bcf7497f93 8005849: JEP 167: Event-Based JVM Tracing
sla
parents: 17134
diff changeset
  4379
  return fetcher.result();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4380
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4381
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4382
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4383
// This does not do anything on Solaris. This is basically a hook for being
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4384
// able to use structured exception handling (thread-local exception filters) on, e.g., Win32.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4385
void os::os_exception_wrapper(java_call_t f, JavaValue* value, methodHandle* method, JavaCallArguments* args, Thread* thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4386
  f(value, method, args, thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4387
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4388
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4389
// This routine may be used by user applications as a "hook" to catch signals.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4390
// The user-defined signal handler must pass unrecognized signals to this
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4391
// routine, and if it returns true (non-zero), then the signal handler must
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4392
// return immediately.  If the flag "abort_if_unrecognized" is true, then this
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4393
// routine will never retun false (zero), but instead will execute a VM panic
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4394
// routine kill the process.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4395
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4396
// If this routine returns false, it is OK to call it again.  This allows
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4397
// the user-defined signal handler to perform checks either before or after
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4398
// the VM performs its own checks.  Naturally, the user code would be making
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4399
// a serious error if it tried to handle an exception (such as a null check
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4400
// or breakpoint) that the VM was generating for its own correct operation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4401
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4402
// This routine may recognize any of the following kinds of signals:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4403
// SIGBUS, SIGSEGV, SIGILL, SIGFPE, BREAK_SIGNAL, SIGPIPE, SIGXFSZ,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4404
// os::Solaris::SIGasync
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4405
// It should be consulted by handlers for any of those signals.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4406
// It explicitly does not recognize os::Solaris::SIGinterrupt
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4407
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4408
// The caller of this routine must pass in the three arguments supplied
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4409
// to the function referred to in the "sa_sigaction" (not the "sa_handler")
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4410
// field of the structure passed to sigaction().  This routine assumes that
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4411
// the sa_flags field passed to sigaction() includes SA_SIGINFO and SA_RESTART.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4412
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4413
// Note that the VM will print warnings if it detects conflicting signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4414
// handlers, unless invoked with the option "-XX:+AllowUserSignalHandlers".
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4415
//
8106
19106a0203fb 6588413: Use -fvisibility=hidden for gcc compiles
coleenp
parents: 7901
diff changeset
  4416
extern "C" JNIEXPORT int
19106a0203fb 6588413: Use -fvisibility=hidden for gcc compiles
coleenp
parents: 7901
diff changeset
  4417
JVM_handle_solaris_signal(int signo, siginfo_t* siginfo, void* ucontext,
19106a0203fb 6588413: Use -fvisibility=hidden for gcc compiles
coleenp
parents: 7901
diff changeset
  4418
                          int abort_if_unrecognized);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4419
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4420
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4421
void signalHandler(int sig, siginfo_t* info, void* ucVoid) {
15743
f708934a12e7 6749267: Signal handler should save/restore errno
hseigel
parents: 15742
diff changeset
  4422
  int orig_errno = errno;  // Preserve errno value over signal handler.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4423
  JVM_handle_solaris_signal(sig, info, ucVoid, true);
15743
f708934a12e7 6749267: Signal handler should save/restore errno
hseigel
parents: 15742
diff changeset
  4424
  errno = orig_errno;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4425
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4426
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4427
/* Do not delete - if guarantee is ever removed,  a signal handler (even empty)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4428
   is needed to provoke threads blocked on IO to return an EINTR
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4429
   Note: this explicitly does NOT call JVM_handle_solaris_signal and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4430
   does NOT participate in signal chaining due to requirement for
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4431
   NOT setting SA_RESTART to make EINTR work. */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4432
extern "C" void sigINTRHandler(int sig, siginfo_t* info, void* ucVoid) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4433
   if (UseSignalChaining) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4434
      struct sigaction *actp = os::Solaris::get_chained_signal_action(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4435
      if (actp && actp->sa_handler) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4436
        vm_exit_during_initialization("Signal chaining detected for VM interrupt signal, try -XX:+UseAltSigs");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4437
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4438
   }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4439
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4440
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4441
// This boolean allows users to forward their own non-matching signals
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4442
// to JVM_handle_solaris_signal, harmlessly.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4443
bool os::Solaris::signal_handlers_are_installed = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4444
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4445
// For signal-chaining
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4446
bool os::Solaris::libjsig_is_loaded = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4447
typedef struct sigaction *(*get_signal_t)(int);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4448
get_signal_t os::Solaris::get_signal_action = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4449
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4450
struct sigaction* os::Solaris::get_chained_signal_action(int sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4451
  struct sigaction *actp = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4452
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4453
  if ((libjsig_is_loaded)  && (sig <= Maxlibjsigsigs)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4454
    // Retrieve the old signal handler from libjsig
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4455
    actp = (*get_signal_action)(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4456
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4457
  if (actp == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4458
    // Retrieve the preinstalled signal handler from jvm
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4459
    actp = get_preinstalled_handler(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4460
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4461
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4462
  return actp;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4463
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4464
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4465
static bool call_chained_handler(struct sigaction *actp, int sig,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4466
                                 siginfo_t *siginfo, void *context) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4467
  // Call the old signal handler
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4468
  if (actp->sa_handler == SIG_DFL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4469
    // It's more reasonable to let jvm treat it as an unexpected exception
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4470
    // instead of taking the default action.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4471
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4472
  } else if (actp->sa_handler != SIG_IGN) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4473
    if ((actp->sa_flags & SA_NODEFER) == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4474
      // automaticlly block the signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4475
      sigaddset(&(actp->sa_mask), sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4476
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4477
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4478
    sa_handler_t hand;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4479
    sa_sigaction_t sa;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4480
    bool siginfo_flag_set = (actp->sa_flags & SA_SIGINFO) != 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4481
    // retrieve the chained handler
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4482
    if (siginfo_flag_set) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4483
      sa = actp->sa_sigaction;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4484
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4485
      hand = actp->sa_handler;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4486
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4487
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4488
    if ((actp->sa_flags & SA_RESETHAND) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4489
      actp->sa_handler = SIG_DFL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4490
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4491
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4492
    // try to honor the signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4493
    sigset_t oset;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4494
    thr_sigsetmask(SIG_SETMASK, &(actp->sa_mask), &oset);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4495
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4496
    // call into the chained handler
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4497
    if (siginfo_flag_set) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4498
      (*sa)(sig, siginfo, context);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4499
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4500
      (*hand)(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4501
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4502
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4503
    // restore the signal mask
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4504
    thr_sigsetmask(SIG_SETMASK, &oset, 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4505
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4506
  // Tell jvm's signal handler the signal is taken care of.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4507
  return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4508
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4509
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4510
bool os::Solaris::chained_handler(int sig, siginfo_t* siginfo, void* context) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4511
  bool chained = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4512
  // signal-chaining
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4513
  if (UseSignalChaining) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4514
    struct sigaction *actp = get_chained_signal_action(sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4515
    if (actp != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4516
      chained = call_chained_handler(actp, sig, siginfo, context);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4517
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4518
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4519
  return chained;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4520
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4521
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4522
struct sigaction* os::Solaris::get_preinstalled_handler(int sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4523
  assert((chainedsigactions != (struct sigaction *)NULL) && (preinstalled_sigs != (int *)NULL) , "signals not yet initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4524
  if (preinstalled_sigs[sig] != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4525
    return &chainedsigactions[sig];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4526
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4527
  return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4528
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4529
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4530
void os::Solaris::save_preinstalled_handler(int sig, struct sigaction& oldAct) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4531
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4532
  assert(sig > 0 && sig <= Maxsignum, "vm signal out of expected range");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4533
  assert((chainedsigactions != (struct sigaction *)NULL) && (preinstalled_sigs != (int *)NULL) , "signals not yet initialized");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4534
  chainedsigactions[sig] = oldAct;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4535
  preinstalled_sigs[sig] = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4536
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4537
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4538
void os::Solaris::set_signal_handler(int sig, bool set_installed, bool oktochain) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4539
  // Check for overwrite.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4540
  struct sigaction oldAct;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4541
  sigaction(sig, (struct sigaction*)NULL, &oldAct);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4542
  void* oldhand = oldAct.sa_sigaction ? CAST_FROM_FN_PTR(void*,  oldAct.sa_sigaction)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4543
                                      : CAST_FROM_FN_PTR(void*,  oldAct.sa_handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4544
  if (oldhand != CAST_FROM_FN_PTR(void*, SIG_DFL) &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4545
      oldhand != CAST_FROM_FN_PTR(void*, SIG_IGN) &&
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4546
      oldhand != CAST_FROM_FN_PTR(void*, signalHandler)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4547
    if (AllowUserSignalHandlers || !set_installed) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4548
      // Do not overwrite; user takes responsibility to forward to us.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4549
      return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4550
    } else if (UseSignalChaining) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4551
      if (oktochain) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4552
        // save the old handler in jvm
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4553
        save_preinstalled_handler(sig, oldAct);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4554
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4555
        vm_exit_during_initialization("Signal chaining not allowed for VM interrupt signal, try -XX:+UseAltSigs.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4556
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4557
      // libjsig also interposes the sigaction() call below and saves the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4558
      // old sigaction on it own.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4559
    } else {
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  4560
      fatal(err_msg("Encountered unexpected pre-existing sigaction handler "
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  4561
                    "%#lx for signal %d.", (long)oldhand, sig));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4562
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4563
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4564
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4565
  struct sigaction sigAct;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4566
  sigfillset(&(sigAct.sa_mask));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4567
  sigAct.sa_handler = SIG_DFL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4568
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4569
  sigAct.sa_sigaction = signalHandler;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4570
  // Handle SIGSEGV on alternate signal stack if
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4571
  // not using stack banging
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4572
  if (!UseStackBanging && sig == SIGSEGV) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4573
    sigAct.sa_flags = SA_SIGINFO | SA_RESTART | SA_ONSTACK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4574
  // Interruptible i/o requires SA_RESTART cleared so EINTR
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4575
  // is returned instead of restarting system calls
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4576
  } else if (sig == os::Solaris::SIGinterrupt()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4577
    sigemptyset(&sigAct.sa_mask);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4578
    sigAct.sa_handler = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4579
    sigAct.sa_flags = SA_SIGINFO;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4580
    sigAct.sa_sigaction = sigINTRHandler;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4581
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4582
    sigAct.sa_flags = SA_SIGINFO | SA_RESTART;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4583
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4584
  os::Solaris::set_our_sigflags(sig, sigAct.sa_flags);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4585
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4586
  sigaction(sig, &sigAct, &oldAct);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4587
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4588
  void* oldhand2 = oldAct.sa_sigaction ? CAST_FROM_FN_PTR(void*, oldAct.sa_sigaction)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4589
                                       : CAST_FROM_FN_PTR(void*, oldAct.sa_handler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4590
  assert(oldhand2 == oldhand, "no concurrent signal handler installation");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4591
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4592
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4593
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4594
#define DO_SIGNAL_CHECK(sig) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4595
  if (!sigismember(&check_signal_done, sig)) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4596
    os::Solaris::check_signal_handler(sig)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4597
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4598
// This method is a periodic task to check for misbehaving JNI applications
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4599
// under CheckJNI, we can add any periodic checks here
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4600
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4601
void os::run_periodic_checks() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4602
  // A big source of grief is hijacking virt. addr 0x0 on Solaris,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4603
  // thereby preventing a NULL checks.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4604
  if(!check_addr0_done) check_addr0_done = check_addr0(tty);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4605
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4606
  if (check_signals == false) return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4607
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4608
  // SEGV and BUS if overridden could potentially prevent
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4609
  // generation of hs*.log in the event of a crash, debugging
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4610
  // such a case can be very challenging, so we absolutely
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4611
  // check for the following for a good measure:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4612
  DO_SIGNAL_CHECK(SIGSEGV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4613
  DO_SIGNAL_CHECK(SIGILL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4614
  DO_SIGNAL_CHECK(SIGFPE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4615
  DO_SIGNAL_CHECK(SIGBUS);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4616
  DO_SIGNAL_CHECK(SIGPIPE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4617
  DO_SIGNAL_CHECK(SIGXFSZ);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4618
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4619
  // ReduceSignalUsage allows the user to override these handlers
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4620
  // see comments at the very top and jvm_solaris.h
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4621
  if (!ReduceSignalUsage) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4622
    DO_SIGNAL_CHECK(SHUTDOWN1_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4623
    DO_SIGNAL_CHECK(SHUTDOWN2_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4624
    DO_SIGNAL_CHECK(SHUTDOWN3_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4625
    DO_SIGNAL_CHECK(BREAK_SIGNAL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4626
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4627
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4628
  // See comments above for using JVM1/JVM2 and UseAltSigs
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4629
  DO_SIGNAL_CHECK(os::Solaris::SIGinterrupt());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4630
  DO_SIGNAL_CHECK(os::Solaris::SIGasync());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4631
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4632
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4633
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4634
typedef int (*os_sigaction_t)(int, const struct sigaction *, struct sigaction *);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4635
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4636
static os_sigaction_t os_sigaction = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4637
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4638
void os::Solaris::check_signal_handler(int sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4639
  char buf[O_BUFLEN];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4640
  address jvmHandler = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4641
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4642
  struct sigaction act;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4643
  if (os_sigaction == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4644
    // only trust the default sigaction, in case it has been interposed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4645
    os_sigaction = (os_sigaction_t)dlsym(RTLD_DEFAULT, "sigaction");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4646
    if (os_sigaction == NULL) return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4647
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4648
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4649
  os_sigaction(sig, (struct sigaction*)NULL, &act);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4650
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4651
  address thisHandler = (act.sa_flags & SA_SIGINFO)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4652
    ? CAST_FROM_FN_PTR(address, act.sa_sigaction)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4653
    : CAST_FROM_FN_PTR(address, act.sa_handler) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4654
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4655
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4656
  switch(sig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4657
    case SIGSEGV:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4658
    case SIGBUS:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4659
    case SIGFPE:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4660
    case SIGPIPE:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4661
    case SIGXFSZ:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4662
    case SIGILL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4663
      jvmHandler = CAST_FROM_FN_PTR(address, signalHandler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4664
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4665
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4666
    case SHUTDOWN1_SIGNAL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4667
    case SHUTDOWN2_SIGNAL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4668
    case SHUTDOWN3_SIGNAL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4669
    case BREAK_SIGNAL:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4670
      jvmHandler = (address)user_handler();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4671
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4672
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4673
    default:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4674
      int intrsig = os::Solaris::SIGinterrupt();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4675
      int asynsig = os::Solaris::SIGasync();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4676
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4677
      if (sig == intrsig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4678
        jvmHandler = CAST_FROM_FN_PTR(address, sigINTRHandler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4679
      } else if (sig == asynsig) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4680
        jvmHandler = CAST_FROM_FN_PTR(address, signalHandler);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4681
      } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4682
        return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4683
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4684
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4685
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4686
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4687
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4688
  if (thisHandler != jvmHandler) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4689
    tty->print("Warning: %s handler ", exception_name(sig, buf, O_BUFLEN));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4690
    tty->print("expected:%s", get_signal_handler_name(jvmHandler, buf, O_BUFLEN));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4691
    tty->print_cr("  found:%s", get_signal_handler_name(thisHandler, buf, O_BUFLEN));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4692
    // No need to check this sig any longer
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4693
    sigaddset(&check_signal_done, sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4694
  } else if(os::Solaris::get_our_sigflags(sig) != 0 && act.sa_flags != os::Solaris::get_our_sigflags(sig)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4695
    tty->print("Warning: %s handler flags ", exception_name(sig, buf, O_BUFLEN));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4696
    tty->print("expected:" PTR32_FORMAT, os::Solaris::get_our_sigflags(sig));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4697
    tty->print_cr("  found:" PTR32_FORMAT, act.sa_flags);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4698
    // No need to check this sig any longer
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4699
    sigaddset(&check_signal_done, sig);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4700
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4701
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4702
  // Print all the signal handler state
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4703
  if (sigismember(&check_signal_done, sig)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4704
    print_signal_handlers(tty, buf, O_BUFLEN);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4705
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4706
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4707
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4708
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4709
void os::Solaris::install_signal_handlers() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4710
  bool libjsigdone = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4711
  signal_handlers_are_installed = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4712
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4713
  // signal-chaining
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4714
  typedef void (*signal_setting_t)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4715
  signal_setting_t begin_signal_setting = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4716
  signal_setting_t end_signal_setting = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4717
  begin_signal_setting = CAST_TO_FN_PTR(signal_setting_t,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4718
                                        dlsym(RTLD_DEFAULT, "JVM_begin_signal_setting"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4719
  if (begin_signal_setting != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4720
    end_signal_setting = CAST_TO_FN_PTR(signal_setting_t,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4721
                                        dlsym(RTLD_DEFAULT, "JVM_end_signal_setting"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4722
    get_signal_action = CAST_TO_FN_PTR(get_signal_t,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4723
                                       dlsym(RTLD_DEFAULT, "JVM_get_signal_action"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4724
    get_libjsig_version = CAST_TO_FN_PTR(version_getting_t,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4725
                                         dlsym(RTLD_DEFAULT, "JVM_get_libjsig_version"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4726
    libjsig_is_loaded = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4727
    if (os::Solaris::get_libjsig_version != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4728
      libjsigversion =  (*os::Solaris::get_libjsig_version)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4729
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4730
    assert(UseSignalChaining, "should enable signal-chaining");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4731
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4732
  if (libjsig_is_loaded) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4733
    // Tell libjsig jvm is setting signal handlers
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4734
    (*begin_signal_setting)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4735
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4736
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4737
  set_signal_handler(SIGSEGV, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4738
  set_signal_handler(SIGPIPE, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4739
  set_signal_handler(SIGXFSZ, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4740
  set_signal_handler(SIGBUS, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4741
  set_signal_handler(SIGILL, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4742
  set_signal_handler(SIGFPE, true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4743
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4744
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4745
  if (os::Solaris::SIGinterrupt() > OLDMAXSIGNUM || os::Solaris::SIGasync() > OLDMAXSIGNUM) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4746
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4747
    // Pre-1.4.1 Libjsig limited to signal chaining signals <= 32 so
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4748
    // can not register overridable signals which might be > 32
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4749
    if (libjsig_is_loaded && libjsigversion <= JSIG_VERSION_1_4_1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4750
    // Tell libjsig jvm has finished setting signal handlers
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4751
      (*end_signal_setting)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4752
      libjsigdone = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4753
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4754
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4755
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4756
  // Never ok to chain our SIGinterrupt
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4757
  set_signal_handler(os::Solaris::SIGinterrupt(), true, false);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4758
  set_signal_handler(os::Solaris::SIGasync(), true, true);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4759
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4760
  if (libjsig_is_loaded && !libjsigdone) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4761
    // Tell libjsig jvm finishes setting signal handlers
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4762
    (*end_signal_setting)();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4763
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4764
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4765
  // We don't activate signal checker if libjsig is in place, we trust ourselves
10561
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4766
  // and if UserSignalHandler is installed all bets are off.
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4767
  // Log that signal checking is off only if -verbose:jni is specified.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4768
  if (CheckJNICalls) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4769
    if (libjsig_is_loaded) {
10561
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4770
      if (PrintJNIResolving) {
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4771
        tty->print_cr("Info: libjsig is activated, all active signal checking is disabled");
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4772
      }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4773
      check_signals = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4774
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4775
    if (AllowUserSignalHandlers) {
10561
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4776
      if (PrintJNIResolving) {
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4777
        tty->print_cr("Info: AllowUserSignalHandlers is activated, all active signal checking is disabled");
bf51fe78a9ad 7051189: Need to suppress info message if -xcheck:jni used with libjsig.so
kevinw
parents: 10494
diff changeset
  4778
      }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4779
      check_signals = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4780
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4781
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4782
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4783
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4784
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4785
void report_error(const char* file_name, int line_no, const char* title, const char* format, ...);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4786
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4787
const char * signames[] = {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4788
  "SIG0",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4789
  "SIGHUP", "SIGINT", "SIGQUIT", "SIGILL", "SIGTRAP",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4790
  "SIGABRT", "SIGEMT", "SIGFPE", "SIGKILL", "SIGBUS",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4791
  "SIGSEGV", "SIGSYS", "SIGPIPE", "SIGALRM", "SIGTERM",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4792
  "SIGUSR1", "SIGUSR2", "SIGCLD", "SIGPWR", "SIGWINCH",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4793
  "SIGURG", "SIGPOLL", "SIGSTOP", "SIGTSTP", "SIGCONT",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4794
  "SIGTTIN", "SIGTTOU", "SIGVTALRM", "SIGPROF", "SIGXCPU",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4795
  "SIGXFSZ", "SIGWAITING", "SIGLWP", "SIGFREEZE", "SIGTHAW",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4796
  "SIGCANCEL", "SIGLOST"
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4797
};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4798
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4799
const char* os::exception_name(int exception_code, char* buf, size_t size) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4800
  if (0 < exception_code && exception_code <= SIGRTMAX) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4801
    // signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4802
    if (exception_code < sizeof(signames)/sizeof(const char*)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4803
       jio_snprintf(buf, size, "%s", signames[exception_code]);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4804
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4805
       jio_snprintf(buf, size, "SIG%d", exception_code);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4806
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4807
    return buf;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4808
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4809
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4810
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4811
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4812
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4813
// (Static) wrappers for the new libthread API
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4814
int_fnP_thread_t_iP_uP_stack_tP_gregset_t os::Solaris::_thr_getstate;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4815
int_fnP_thread_t_i_gregset_t os::Solaris::_thr_setstate;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4816
int_fnP_thread_t_i os::Solaris::_thr_setmutator;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4817
int_fnP_thread_t os::Solaris::_thr_suspend_mutator;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4818
int_fnP_thread_t os::Solaris::_thr_continue_mutator;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4819
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4820
// (Static) wrapper for getisax(2) call.
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4821
os::Solaris::getisax_func_t os::Solaris::_getisax = 0;
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4822
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4823
// (Static) wrappers for the liblgrp API
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4824
os::Solaris::lgrp_home_func_t os::Solaris::_lgrp_home;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4825
os::Solaris::lgrp_init_func_t os::Solaris::_lgrp_init;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4826
os::Solaris::lgrp_fini_func_t os::Solaris::_lgrp_fini;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4827
os::Solaris::lgrp_root_func_t os::Solaris::_lgrp_root;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4828
os::Solaris::lgrp_children_func_t os::Solaris::_lgrp_children;
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  4829
os::Solaris::lgrp_resources_func_t os::Solaris::_lgrp_resources;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4830
os::Solaris::lgrp_nlgrps_func_t os::Solaris::_lgrp_nlgrps;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4831
os::Solaris::lgrp_cookie_stale_func_t os::Solaris::_lgrp_cookie_stale;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4832
os::Solaris::lgrp_cookie_t os::Solaris::_lgrp_cookie = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4833
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4834
// (Static) wrapper for meminfo() call.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4835
os::Solaris::meminfo_func_t os::Solaris::_meminfo = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4836
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4837
static address resolve_symbol_lazy(const char* name) {
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4838
  address addr = (address) dlsym(RTLD_DEFAULT, name);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4839
  if(addr == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4840
    // RTLD_DEFAULT was not defined on some early versions of 2.5.1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4841
    addr = (address) dlsym(RTLD_NEXT, name);
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4842
  }
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4843
  return addr;
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4844
}
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4845
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4846
static address resolve_symbol(const char* name) {
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4847
  address addr = resolve_symbol_lazy(name);
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4848
  if(addr == NULL) {
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  4849
    fatal(dlerror());
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4850
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4851
  return addr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4852
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4853
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4854
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4855
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4856
// isT2_libthread()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4857
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4858
// Routine to determine if we are currently using the new T2 libthread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4859
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4860
// We determine if we are using T2 by reading /proc/self/lstatus and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4861
// looking for a thread with the ASLWP bit set.  If we find this status
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4862
// bit set, we must assume that we are NOT using T2.  The T2 team
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4863
// has approved this algorithm.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4864
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4865
// We need to determine if we are running with the new T2 libthread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4866
// since setting native thread priorities is handled differently
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4867
// when using this library.  All threads created using T2 are bound
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4868
// threads. Calling thr_setprio is meaningless in this case.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4869
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4870
bool isT2_libthread() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4871
  static prheader_t * lwpArray = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4872
  static int lwpSize = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4873
  static int lwpFile = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4874
  lwpstatus_t * that;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4875
  char lwpName [128];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4876
  bool isT2 = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4877
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4878
#define ADR(x)  ((uintptr_t)(x))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4879
#define LWPINDEX(ary,ix)   ((lwpstatus_t *)(((ary)->pr_entsize * (ix)) + (ADR((ary) + 1))))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4880
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  4881
  lwpFile = ::open("/proc/self/lstatus", O_RDONLY, 0);
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4882
  if (lwpFile < 0) {
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4883
      if (ThreadPriorityVerbose) warning ("Couldn't open /proc/self/lstatus\n");
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4884
      return false;
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4885
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4886
  lwpSize = 16*1024;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4887
  for (;;) {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  4888
    ::lseek64 (lwpFile, 0, SEEK_SET);
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  4889
    lwpArray = (prheader_t *)NEW_C_HEAP_ARRAY(char, lwpSize, mtInternal);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  4890
    if (::read(lwpFile, lwpArray, lwpSize) < 0) {
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4891
      if (ThreadPriorityVerbose) warning("Error reading /proc/self/lstatus\n");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4892
      break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4893
    }
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4894
    if ((lwpArray->pr_nent * lwpArray->pr_entsize) <= lwpSize) {
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4895
       // We got a good snapshot - now iterate over the list.
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4896
      int aslwpcount = 0;
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4897
      for (int i = 0; i < lwpArray->pr_nent; i++ ) {
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4898
        that = LWPINDEX(lwpArray,i);
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4899
        if (that->pr_flags & PR_ASLWP) {
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4900
          aslwpcount++;
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4901
        }
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4902
      }
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4903
      if (aslwpcount == 0) isT2 = true;
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4904
      break;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4905
    }
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4906
    lwpSize = lwpArray->pr_nent * lwpArray->pr_entsize;
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  4907
    FREE_C_HEAP_ARRAY(char, lwpArray, mtInternal);  // retry.
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  4908
  }
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  4909
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  4910
  FREE_C_HEAP_ARRAY(char, lwpArray, mtInternal);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  4911
  ::close (lwpFile);
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4912
  if (ThreadPriorityVerbose) {
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4913
    if (isT2) tty->print_cr("We are running with a T2 libthread\n");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4914
    else tty->print_cr("We are not running with a T2 libthread\n");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4915
  }
336
53887b25f9ae 6671882: memory access after free in solaris/vm/os_solaris.cpp
xlu
parents: 252
diff changeset
  4916
  return isT2;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4917
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4918
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4919
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4920
void os::Solaris::libthread_init() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4921
  address func = (address)dlsym(RTLD_DEFAULT, "_thr_suspend_allmutators");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4922
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4923
  // Determine if we are running with the new T2 libthread
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4924
  os::Solaris::set_T2_libthread(isT2_libthread());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4925
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4926
  lwp_priocntl_init();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4927
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4928
  // RTLD_DEFAULT was not defined on some early versions of 5.5.1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4929
  if(func == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4930
    func = (address) dlsym(RTLD_NEXT, "_thr_suspend_allmutators");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4931
    // Guarantee that this VM is running on an new enough OS (5.6 or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4932
    // later) that it will have a new enough libthread.so.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4933
    guarantee(func != NULL, "libthread.so is too old.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4934
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4935
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4936
  // Initialize the new libthread getstate API wrappers
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4937
  func = resolve_symbol("thr_getstate");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4938
  os::Solaris::set_thr_getstate(CAST_TO_FN_PTR(int_fnP_thread_t_iP_uP_stack_tP_gregset_t, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4939
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4940
  func = resolve_symbol("thr_setstate");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4941
  os::Solaris::set_thr_setstate(CAST_TO_FN_PTR(int_fnP_thread_t_i_gregset_t, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4942
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4943
  func = resolve_symbol("thr_setmutator");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4944
  os::Solaris::set_thr_setmutator(CAST_TO_FN_PTR(int_fnP_thread_t_i, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4945
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4946
  func = resolve_symbol("thr_suspend_mutator");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4947
  os::Solaris::set_thr_suspend_mutator(CAST_TO_FN_PTR(int_fnP_thread_t, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4948
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4949
  func = resolve_symbol("thr_continue_mutator");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4950
  os::Solaris::set_thr_continue_mutator(CAST_TO_FN_PTR(int_fnP_thread_t, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4951
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4952
  int size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4953
  void (*handler_info_func)(address *, int *);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4954
  handler_info_func = CAST_TO_FN_PTR(void (*)(address *, int *), resolve_symbol("thr_sighndlrinfo"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4955
  handler_info_func(&handler_start, &size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4956
  handler_end = handler_start + size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4957
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4958
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4959
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4960
int_fnP_mutex_tP os::Solaris::_mutex_lock;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4961
int_fnP_mutex_tP os::Solaris::_mutex_trylock;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4962
int_fnP_mutex_tP os::Solaris::_mutex_unlock;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4963
int_fnP_mutex_tP_i_vP os::Solaris::_mutex_init;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4964
int_fnP_mutex_tP os::Solaris::_mutex_destroy;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4965
int os::Solaris::_mutex_scope = USYNC_THREAD;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4966
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4967
int_fnP_cond_tP_mutex_tP_timestruc_tP os::Solaris::_cond_timedwait;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4968
int_fnP_cond_tP_mutex_tP os::Solaris::_cond_wait;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4969
int_fnP_cond_tP os::Solaris::_cond_signal;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4970
int_fnP_cond_tP os::Solaris::_cond_broadcast;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4971
int_fnP_cond_tP_i_vP os::Solaris::_cond_init;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4972
int_fnP_cond_tP os::Solaris::_cond_destroy;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4973
int os::Solaris::_cond_scope = USYNC_THREAD;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4974
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4975
void os::Solaris::synchronization_init() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4976
  if(UseLWPSynchronization) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4977
    os::Solaris::set_mutex_lock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("_lwp_mutex_lock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4978
    os::Solaris::set_mutex_trylock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("_lwp_mutex_trylock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4979
    os::Solaris::set_mutex_unlock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("_lwp_mutex_unlock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4980
    os::Solaris::set_mutex_init(lwp_mutex_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4981
    os::Solaris::set_mutex_destroy(lwp_mutex_destroy);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4982
    os::Solaris::set_mutex_scope(USYNC_THREAD);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4983
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4984
    os::Solaris::set_cond_timedwait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP_timestruc_tP, resolve_symbol("_lwp_cond_timedwait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4985
    os::Solaris::set_cond_wait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP, resolve_symbol("_lwp_cond_wait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4986
    os::Solaris::set_cond_signal(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("_lwp_cond_signal")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4987
    os::Solaris::set_cond_broadcast(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("_lwp_cond_broadcast")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4988
    os::Solaris::set_cond_init(lwp_cond_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4989
    os::Solaris::set_cond_destroy(lwp_cond_destroy);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4990
    os::Solaris::set_cond_scope(USYNC_THREAD);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4991
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4992
  else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4993
    os::Solaris::set_mutex_scope(USYNC_THREAD);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4994
    os::Solaris::set_cond_scope(USYNC_THREAD);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4995
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4996
    if(UsePthreads) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4997
      os::Solaris::set_mutex_lock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("pthread_mutex_lock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4998
      os::Solaris::set_mutex_trylock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("pthread_mutex_trylock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  4999
      os::Solaris::set_mutex_unlock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("pthread_mutex_unlock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5000
      os::Solaris::set_mutex_init(pthread_mutex_default_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5001
      os::Solaris::set_mutex_destroy(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("pthread_mutex_destroy")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5002
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5003
      os::Solaris::set_cond_timedwait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP_timestruc_tP, resolve_symbol("pthread_cond_timedwait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5004
      os::Solaris::set_cond_wait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP, resolve_symbol("pthread_cond_wait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5005
      os::Solaris::set_cond_signal(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("pthread_cond_signal")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5006
      os::Solaris::set_cond_broadcast(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("pthread_cond_broadcast")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5007
      os::Solaris::set_cond_init(pthread_cond_default_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5008
      os::Solaris::set_cond_destroy(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("pthread_cond_destroy")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5009
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5010
    else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5011
      os::Solaris::set_mutex_lock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("mutex_lock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5012
      os::Solaris::set_mutex_trylock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("mutex_trylock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5013
      os::Solaris::set_mutex_unlock(CAST_TO_FN_PTR(int_fnP_mutex_tP, resolve_symbol("mutex_unlock")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5014
      os::Solaris::set_mutex_init(::mutex_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5015
      os::Solaris::set_mutex_destroy(::mutex_destroy);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5016
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5017
      os::Solaris::set_cond_timedwait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP_timestruc_tP, resolve_symbol("cond_timedwait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5018
      os::Solaris::set_cond_wait(CAST_TO_FN_PTR(int_fnP_cond_tP_mutex_tP, resolve_symbol("cond_wait")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5019
      os::Solaris::set_cond_signal(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("cond_signal")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5020
      os::Solaris::set_cond_broadcast(CAST_TO_FN_PTR(int_fnP_cond_tP, resolve_symbol("cond_broadcast")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5021
      os::Solaris::set_cond_init(::cond_init);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5022
      os::Solaris::set_cond_destroy(::cond_destroy);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5023
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5024
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5025
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5026
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5027
bool os::Solaris::liblgrp_init() {
975
ad7da100aa6a 6720130: NUMA allocator: The linux version should search for libnuma.so.1
iveresov
parents: 391
diff changeset
  5028
  void *handle = dlopen("liblgrp.so.1", RTLD_LAZY);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5029
  if (handle != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5030
    os::Solaris::set_lgrp_home(CAST_TO_FN_PTR(lgrp_home_func_t, dlsym(handle, "lgrp_home")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5031
    os::Solaris::set_lgrp_init(CAST_TO_FN_PTR(lgrp_init_func_t, dlsym(handle, "lgrp_init")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5032
    os::Solaris::set_lgrp_fini(CAST_TO_FN_PTR(lgrp_fini_func_t, dlsym(handle, "lgrp_fini")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5033
    os::Solaris::set_lgrp_root(CAST_TO_FN_PTR(lgrp_root_func_t, dlsym(handle, "lgrp_root")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5034
    os::Solaris::set_lgrp_children(CAST_TO_FN_PTR(lgrp_children_func_t, dlsym(handle, "lgrp_children")));
391
f889070a8684 6697534: Premature GC and invalid lgrp selection with NUMA-aware allocator.
iveresov
parents: 388
diff changeset
  5035
    os::Solaris::set_lgrp_resources(CAST_TO_FN_PTR(lgrp_resources_func_t, dlsym(handle, "lgrp_resources")));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5036
    os::Solaris::set_lgrp_nlgrps(CAST_TO_FN_PTR(lgrp_nlgrps_func_t, dlsym(handle, "lgrp_nlgrps")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5037
    os::Solaris::set_lgrp_cookie_stale(CAST_TO_FN_PTR(lgrp_cookie_stale_func_t,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5038
                                       dlsym(handle, "lgrp_cookie_stale")));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5039
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5040
    lgrp_cookie_t c = lgrp_init(LGRP_VIEW_CALLER);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5041
    set_lgrp_cookie(c);
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5042
    return true;
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5043
  }
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5044
  return false;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5045
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5046
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5047
void os::Solaris::misc_sym_init() {
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5048
  address func;
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5049
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5050
  // getisax
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5051
  func = resolve_symbol_lazy("getisax");
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5052
  if (func != NULL) {
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5053
    os::Solaris::_getisax = CAST_TO_FN_PTR(getisax_func_t, func);
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5054
  }
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5055
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5056
  // meminfo
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5057
  func = resolve_symbol_lazy("meminfo");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5058
  if (func != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5059
    os::Solaris::set_meminfo(CAST_TO_FN_PTR(meminfo_func_t, func));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5060
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5061
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5062
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5063
uint_t os::Solaris::getisax(uint32_t* array, uint_t n) {
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5064
  assert(_getisax != NULL, "_getisax not set");
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5065
  return _getisax(array, n);
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5066
}
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5067
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5068
// int pset_getloadavg(psetid_t pset, double loadavg[], int nelem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5069
typedef long (*pset_getloadavg_type)(psetid_t pset, double loadavg[], int nelem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5070
static pset_getloadavg_type pset_getloadavg_ptr = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5071
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5072
void init_pset_getloadavg_ptr(void) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5073
  pset_getloadavg_ptr =
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5074
    (pset_getloadavg_type)dlsym(RTLD_DEFAULT, "pset_getloadavg");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5075
  if (PrintMiscellaneous && Verbose && pset_getloadavg_ptr == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5076
    warning("pset_getloadavg function not found");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5077
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5078
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5079
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5080
int os::Solaris::_dev_zero_fd = -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5081
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5082
// this is called _before_ the global arguments have been parsed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5083
void os::init(void) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5084
  _initial_pid = getpid();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5085
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5086
  max_hrtime = first_hrtime = gethrtime();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5087
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5088
  init_random(1234567);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5089
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5090
  page_size = sysconf(_SC_PAGESIZE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5091
  if (page_size == -1)
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  5092
    fatal(err_msg("os_solaris.cpp: os::init: sysconf failed (%s)",
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  5093
                  strerror(errno)));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5094
  init_page_sizes((size_t) page_size);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5095
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5096
  Solaris::initialize_system_info();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5097
2253
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5098
  // Initialize misc. symbols as soon as possible, so we can use them
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5099
  // if we need them.
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5100
  Solaris::misc_sym_init();
30268d00878e 6812587: Use auxv to determine SPARC hardware features on Solaris
twisti
parents: 2012
diff changeset
  5101
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5102
  int fd = ::open("/dev/zero", O_RDWR);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5103
  if (fd < 0) {
5403
6b0dd9c75dde 6888954: argument formatting for assert() and friends
jcoomes
parents: 5237
diff changeset
  5104
    fatal(err_msg("os::init: cannot open /dev/zero (%s)", strerror(errno)));
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5105
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5106
    Solaris::set_dev_zero_fd(fd);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5107
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5108
    // Close on exec, child won't inherit.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5109
    fcntl(fd, F_SETFD, FD_CLOEXEC);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5110
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5111
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5112
  clock_tics_per_sec = CLK_TCK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5113
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5114
  // check if dladdr1() exists; dladdr1 can provide more information than
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5115
  // dladdr for os::dll_address_to_function_name. It comes with SunOS 5.9
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5116
  // and is available on linker patches for 5.7 and 5.8.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5117
  // libdl.so must have been loaded, this call is just an entry lookup
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5118
  void * hdl = dlopen("libdl.so", RTLD_NOW);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5119
  if (hdl)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5120
    dladdr1_func = CAST_TO_FN_PTR(dladdr1_func_type, dlsym(hdl, "dladdr1"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5121
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5122
  // (Solaris only) this switches to calls that actually do locking.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5123
  ThreadCritical::initialize();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5124
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5125
  main_thread = thr_self();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5126
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5127
  // Constant minimum stack size allowed. It must be at least
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5128
  // the minimum of what the OS supports (thr_min_stack()), and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5129
  // enough to allow the thread to get to user bytecode execution.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5130
  Solaris::min_stack_allowed = MAX2(thr_min_stack(), Solaris::min_stack_allowed);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5131
  // If the pagesize of the VM is greater than 8K determine the appropriate
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5132
  // number of initial guard pages.  The user can change this with the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5133
  // command line arguments, if needed.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5134
  if (vm_page_size() > 8*K) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5135
    StackYellowPages = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5136
    StackRedPages = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5137
    StackShadowPages = round_to((StackShadowPages*8*K), vm_page_size()) / vm_page_size();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5138
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5139
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5140
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5141
// To install functions for atexit system call
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5142
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5143
  static void perfMemory_exit_helper() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5144
    perfMemory_exit();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5145
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5146
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5147
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5148
// this is called _after_ the global arguments have been parsed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5149
jint os::init_2(void) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5150
  // try to enable extended file IO ASAP, see 6431278
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5151
  os::Solaris::try_enable_extended_io();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5152
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5153
  // Allocate a single page and mark it as readable for safepoint polling.  Also
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5154
  // use this first mmap call to check support for MAP_ALIGN.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5155
  address polling_page = (address)Solaris::mmap_chunk((char*)page_size,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5156
                                                      page_size,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5157
                                                      MAP_PRIVATE | MAP_ALIGN,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5158
                                                      PROT_READ);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5159
  if (polling_page == NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5160
    has_map_align = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5161
    polling_page = (address)Solaris::mmap_chunk(NULL, page_size, MAP_PRIVATE,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5162
                                                PROT_READ);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5163
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5164
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5165
  os::set_polling_page(polling_page);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5166
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5167
#ifndef PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5168
  if( Verbose && PrintMiscellaneous )
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5169
    tty->print("[SafePoint Polling address: " INTPTR_FORMAT "]\n", (intptr_t)polling_page);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5170
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5171
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5172
  if (!UseMembar) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5173
    address mem_serialize_page = (address)Solaris::mmap_chunk( NULL, page_size, MAP_PRIVATE, PROT_READ | PROT_WRITE );
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5174
    guarantee( mem_serialize_page != NULL, "mmap Failed for memory serialize page");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5175
    os::set_memory_serialize_page( mem_serialize_page );
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5176
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5177
#ifndef PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5178
    if(Verbose && PrintMiscellaneous)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5179
      tty->print("[Memory Serialize  Page address: " INTPTR_FORMAT "]\n", (intptr_t)mem_serialize_page);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5180
#endif
19986
33d188c66ed9 8010722: assert: failed: heap size is too big for compressed oops
tschatzl
parents: 19546
diff changeset
  5181
  }
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5182
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5183
  // Check minimum allowable stack size for thread creation and to initialize
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5184
  // the java system classes, including StackOverflowError - depends on page
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5185
  // size.  Add a page for compiler2 recursion in main thread.
6964
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5186
  // Add in 2*BytesPerWord times page size to account for VM stack during
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5187
  // class initialization depending on 32 or 64 bit VM.
6964
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5188
  os::Solaris::min_stack_allowed = MAX2(os::Solaris::min_stack_allowed,
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5189
            (size_t)(StackYellowPages+StackRedPages+StackShadowPages+
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5190
                    2*BytesPerWord COMPILER2_PRESENT(+1)) * page_size);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5191
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5192
  size_t threadStackSizeInBytes = ThreadStackSize * K;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5193
  if (threadStackSizeInBytes != 0 &&
6964
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5194
    threadStackSizeInBytes < os::Solaris::min_stack_allowed) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5195
    tty->print_cr("\nThe stack size specified is too small, Specify at least %dk",
6964
6e45ffa3bccf 6983240: guarantee((Solaris::min_stack_allowed >= (StackYellowPages+StackRedPages...) wrong
coleenp
parents: 6962
diff changeset
  5196
                  os::Solaris::min_stack_allowed/K);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5197
    return JNI_ERR;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5198
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5199
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5200
  // For 64kbps there will be a 64kb page size, which makes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5201
  // the usable default stack size quite a bit less.  Increase the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5202
  // stack for 64kb (or any > than 8kb) pages, this increases
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5203
  // virtual memory fragmentation (since we're not creating the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5204
  // stack on a power of 2 boundary.  The real fix for this
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5205
  // should be to fix the guard page mechanism.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5206
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5207
  if (vm_page_size() > 8*K) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5208
      threadStackSizeInBytes = (threadStackSizeInBytes != 0)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5209
         ? threadStackSizeInBytes +
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5210
           ((StackYellowPages + StackRedPages) * vm_page_size())
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5211
         : 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5212
      ThreadStackSize = threadStackSizeInBytes/K;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5213
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5214
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5215
  // Make the stack size a multiple of the page size so that
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5216
  // the yellow/red zones can be guarded.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5217
  JavaThread::set_stack_size_at_create(round_to(threadStackSizeInBytes,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5218
        vm_page_size()));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5219
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5220
  Solaris::libthread_init();
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5221
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5222
  if (UseNUMA) {
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5223
    if (!Solaris::liblgrp_init()) {
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5224
      UseNUMA = false;
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5225
    } else {
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5226
      size_t lgrp_limit = os::numa_get_groups_num();
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  5227
      int *lgrp_ids = NEW_C_HEAP_ARRAY(int, lgrp_limit, mtInternal);
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5228
      size_t lgrp_num = os::numa_get_leaf_groups(lgrp_ids, lgrp_limit);
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  5229
      FREE_C_HEAP_ARRAY(int, lgrp_ids, mtInternal);
1615
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5230
      if (lgrp_num < 2) {
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5231
        // There's only one locality group, disable NUMA.
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5232
        UseNUMA = false;
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5233
      }
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5234
    }
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5235
    if (!UseNUMA && ForceNUMA) {
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5236
      UseNUMA = true;
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5237
    }
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5238
  }
b46d9f19bde2 6779436: NUMA allocator: libnuma expects certain size of the buffer in numa_node_to_cpus()
iveresov
parents: 1558
diff changeset
  5239
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5240
  Solaris::signal_sets_init();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5241
  Solaris::init_signal_mem();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5242
  Solaris::install_signal_handlers();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5243
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5244
  if (libjsigversion < JSIG_VERSION_1_4_1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5245
    Maxlibjsigsigs = OLDMAXSIGNUM;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5246
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5247
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5248
  // initialize synchronization primitives to use either thread or
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5249
  // lwp synchronization (controlled by UseLWPSynchronization)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5250
  Solaris::synchronization_init();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5251
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5252
  if (MaxFDLimit) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5253
    // set the number of file descriptors to max. print out error
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5254
    // if getrlimit/setrlimit fails but continue regardless.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5255
    struct rlimit nbr_files;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5256
    int status = getrlimit(RLIMIT_NOFILE, &nbr_files);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5257
    if (status != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5258
      if (PrintMiscellaneous && (Verbose || WizardMode))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5259
        perror("os::init_2 getrlimit failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5260
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5261
      nbr_files.rlim_cur = nbr_files.rlim_max;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5262
      status = setrlimit(RLIMIT_NOFILE, &nbr_files);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5263
      if (status != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5264
        if (PrintMiscellaneous && (Verbose || WizardMode))
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5265
          perror("os::init_2 setrlimit failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5266
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5267
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5268
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5269
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5270
  // Calculate theoretical max. size of Threads to guard gainst
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5271
  // artifical out-of-memory situations, where all available address-
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5272
  // space has been reserved by thread stacks. Default stack size is 1Mb.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5273
  size_t pre_thread_stack_size = (JavaThread::stack_size_at_create()) ?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5274
    JavaThread::stack_size_at_create() : (1*K*K);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5275
  assert(pre_thread_stack_size != 0, "Must have a stack");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5276
  // Solaris has a maximum of 4Gb of user programs. Calculate the thread limit when
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5277
  // we should start doing Virtual Memory banging. Currently when the threads will
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5278
  // have used all but 200Mb of space.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5279
  size_t max_address_space = ((unsigned int)4 * K * K * K) - (200 * K * K);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5280
  Solaris::_os_thread_limit = max_address_space / pre_thread_stack_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5281
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5282
  // at-exit methods are called in the reverse order of their registration.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5283
  // In Solaris 7 and earlier, atexit functions are called on return from
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5284
  // main or as a result of a call to exit(3C). There can be only 32 of
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5285
  // these functions registered and atexit() does not set errno. In Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5286
  // 8 and later, there is no limit to the number of functions registered
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5287
  // and atexit() sets errno. In addition, in Solaris 8 and later, atexit
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5288
  // functions are called upon dlclose(3DL) in addition to return from main
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5289
  // and exit(3C).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5290
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5291
  if (PerfAllowAtExitRegistration) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5292
    // only register atexit functions if PerfAllowAtExitRegistration is set.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5293
    // atexit functions can be delayed until process exit time, which
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5294
    // can be problematic for embedded VM situations. Embedded VMs should
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5295
    // call DestroyJavaVM() to assure that VM resources are released.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5296
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5297
    // note: perfMemory_exit_helper atexit function may be removed in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5298
    // the future if the appropriate cleanup code can be added to the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5299
    // VM_Exit VMOperation's doit method.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5300
    if (atexit(perfMemory_exit_helper) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5301
      warning("os::init2 atexit(perfMemory_exit_helper) failed");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5302
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5303
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5304
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5305
  // Init pset_loadavg function pointer
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5306
  init_pset_getloadavg_ptr();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5307
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5308
  return JNI_OK;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5309
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5310
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5311
void os::init_3(void) {
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5312
  return;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5313
}
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5314
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5315
// Mark the polling page as unreadable
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5316
void os::make_polling_page_unreadable(void) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5317
  if( mprotect((char *)_polling_page, page_size, PROT_NONE) != 0 )
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5318
    fatal("Could not disable polling page");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5319
};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5320
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5321
// Mark the polling page as readable
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5322
void os::make_polling_page_readable(void) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5323
  if( mprotect((char *)_polling_page, page_size, PROT_READ) != 0 )
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5324
    fatal("Could not enable polling page");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5325
};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5326
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5327
// OS interface.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5328
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5329
bool os::check_heap(bool force) { return true; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5330
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5331
typedef int (*vsnprintf_t)(char* buf, size_t count, const char* fmt, va_list argptr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5332
static vsnprintf_t sol_vsnprintf = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5333
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5334
int local_vsnprintf(char* buf, size_t count, const char* fmt, va_list argptr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5335
  if (!sol_vsnprintf) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5336
    //search  for the named symbol in the objects that were loaded after libjvm
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5337
    void* where = RTLD_NEXT;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5338
    if ((sol_vsnprintf = CAST_TO_FN_PTR(vsnprintf_t, dlsym(where, "__vsnprintf"))) == NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5339
        sol_vsnprintf = CAST_TO_FN_PTR(vsnprintf_t, dlsym(where, "vsnprintf"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5340
    if (!sol_vsnprintf){
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5341
      //search  for the named symbol in the objects that were loaded before libjvm
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5342
      where = RTLD_DEFAULT;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5343
      if ((sol_vsnprintf = CAST_TO_FN_PTR(vsnprintf_t, dlsym(where, "__vsnprintf"))) == NULL)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5344
        sol_vsnprintf = CAST_TO_FN_PTR(vsnprintf_t, dlsym(where, "vsnprintf"));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5345
      assert(sol_vsnprintf != NULL, "vsnprintf not found");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5346
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5347
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5348
  return (*sol_vsnprintf)(buf, count, fmt, argptr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5349
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5350
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5351
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5352
// Is a (classpath) directory empty?
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5353
bool os::dir_is_empty(const char* path) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5354
  DIR *dir = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5355
  struct dirent *ptr;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5356
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5357
  dir = opendir(path);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5358
  if (dir == NULL) return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5359
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5360
  /* Scan the directory */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5361
  bool result = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5362
  char buf[sizeof(struct dirent) + MAX_PATH];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5363
  struct dirent *dbuf = (struct dirent *) buf;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5364
  while (result && (ptr = readdir(dir, dbuf)) != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5365
    if (strcmp(ptr->d_name, ".") != 0 && strcmp(ptr->d_name, "..") != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5366
      result = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5367
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5368
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5369
  closedir(dir);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5370
  return result;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5371
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5372
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5373
// This code originates from JDK's sysOpen and open64_w
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5374
// from src/solaris/hpi/src/system_md.c
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5375
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5376
#ifndef O_DELETE
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5377
#define O_DELETE 0x10000
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5378
#endif
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5379
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5380
// Open a file. Unlink the file immediately after open returns
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5381
// if the specified oflag has the O_DELETE flag set.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5382
// O_DELETE is used only in j2se/src/share/native/java/util/zip/ZipFile.c
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5383
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5384
int os::open(const char *path, int oflag, int mode) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5385
  if (strlen(path) > MAX_PATH - 1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5386
    errno = ENAMETOOLONG;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5387
    return -1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5388
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5389
  int fd;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5390
  int o_delete = (oflag & O_DELETE);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5391
  oflag = oflag & ~O_DELETE;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5392
7721
8fae37350972 7009975: Large file support broken in hs20-b04
alanb
parents: 7692
diff changeset
  5393
  fd = ::open64(path, oflag, mode);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5394
  if (fd == -1) return -1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5395
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5396
  //If the open succeeded, the file might still be a directory
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5397
  {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5398
    struct stat64 buf64;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5399
    int ret = ::fstat64(fd, &buf64);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5400
    int st_mode = buf64.st_mode;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5401
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5402
    if (ret != -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5403
      if ((st_mode & S_IFMT) == S_IFDIR) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5404
        errno = EISDIR;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5405
        ::close(fd);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5406
        return -1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5407
      }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5408
    } else {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5409
      ::close(fd);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5410
      return -1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5411
    }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5412
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5413
    /*
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5414
     * 32-bit Solaris systems suffer from:
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5415
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5416
     * - an historical default soft limit of 256 per-process file
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5417
     *   descriptors that is too low for many Java programs.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5418
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5419
     * - a design flaw where file descriptors created using stdio
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5420
     *   fopen must be less than 256, _even_ when the first limit above
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5421
     *   has been raised.  This can cause calls to fopen (but not calls to
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5422
     *   open, for example) to fail mysteriously, perhaps in 3rd party
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5423
     *   native code (although the JDK itself uses fopen).  One can hardly
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5424
     *   criticize them for using this most standard of all functions.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5425
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5426
     * We attempt to make everything work anyways by:
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5427
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5428
     * - raising the soft limit on per-process file descriptors beyond
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5429
     *   256
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5430
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5431
     * - As of Solaris 10u4, we can request that Solaris raise the 256
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5432
     *   stdio fopen limit by calling function enable_extended_FILE_stdio.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5433
     *   This is done in init_2 and recorded in enabled_extended_FILE_stdio
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5434
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5435
     * - If we are stuck on an old (pre 10u4) Solaris system, we can
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5436
     *   workaround the bug by remapping non-stdio file descriptors below
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5437
     *   256 to ones beyond 256, which is done below.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5438
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5439
     * See:
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5440
     * 1085341: 32-bit stdio routines should support file descriptors >255
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5441
     * 6533291: Work around 32-bit Solaris stdio limit of 256 open files
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5442
     * 6431278: Netbeans crash on 32 bit Solaris: need to call
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5443
     *          enable_extended_FILE_stdio() in VM initialisation
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5444
     * Giri Mandalika's blog
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5445
     * http://technopark02.blogspot.com/2005_05_01_archive.html
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5446
     */
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5447
#ifndef  _LP64
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5448
     if ((!enabled_extended_FILE_stdio) && fd < 256) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5449
         int newfd = ::fcntl(fd, F_DUPFD, 256);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5450
         if (newfd != -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5451
             ::close(fd);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5452
             fd = newfd;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5453
         }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5454
     }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5455
#endif // 32-bit Solaris
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5456
    /*
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5457
     * All file descriptors that are opened in the JVM and not
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5458
     * specifically destined for a subprocess should have the
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5459
     * close-on-exec flag set.  If we don't set it, then careless 3rd
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5460
     * party native code might fork and exec without closing all
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5461
     * appropriate file descriptors (e.g. as we do in closeDescriptors in
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5462
     * UNIXProcess.c), and this in turn might:
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5463
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5464
     * - cause end-of-file to fail to be detected on some file
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5465
     *   descriptors, resulting in mysterious hangs, or
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5466
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5467
     * - might cause an fopen in the subprocess to fail on a system
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5468
     *   suffering from bug 1085341.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5469
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5470
     * (Yes, the default setting of the close-on-exec flag is a Unix
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5471
     * design flaw)
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5472
     *
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5473
     * See:
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5474
     * 1085341: 32-bit stdio routines should support file descriptors >255
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5475
     * 4843136: (process) pipe file descriptor from Runtime.exec not being closed
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5476
     * 6339493: (process) Runtime.exec does not close all file descriptors on Solaris 9
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5477
     */
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5478
#ifdef FD_CLOEXEC
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5479
    {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5480
        int flags = ::fcntl(fd, F_GETFD);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5481
        if (flags != -1)
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5482
            ::fcntl(fd, F_SETFD, flags | FD_CLOEXEC);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5483
    }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5484
#endif
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5485
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5486
  if (o_delete != 0) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5487
    ::unlink(path);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5488
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5489
  return fd;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5490
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5491
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5492
// create binary file, rewriting existing file if required
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5493
int os::create_binary_file(const char* path, bool rewrite_existing) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5494
  int oflags = O_WRONLY | O_CREAT;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5495
  if (!rewrite_existing) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5496
    oflags |= O_EXCL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5497
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5498
  return ::open64(path, oflags, S_IREAD | S_IWRITE);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5499
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5500
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5501
// return current position of file pointer
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5502
jlong os::current_file_offset(int fd) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5503
  return (jlong)::lseek64(fd, (off64_t)0, SEEK_CUR);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5504
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5505
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5506
// move file pointer to the specified offset
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5507
jlong os::seek_to_file_offset(int fd, jlong offset) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5508
  return (jlong)::lseek64(fd, (off64_t)offset, SEEK_SET);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5509
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5510
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5511
jlong os::lseek(int fd, jlong offset, int whence) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5512
  return (jlong) ::lseek64(fd, offset, whence);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5513
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5514
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5515
char * os::native_path(char *path) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5516
  return path;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5517
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5518
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5519
int os::ftruncate(int fd, jlong length) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5520
  return ::ftruncate64(fd, length);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5521
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5522
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5523
int os::fsync(int fd)  {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5524
  RESTARTABLE_RETURN_INT(::fsync(fd));
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5525
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5526
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5527
int os::available(int fd, jlong *bytes) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5528
  jlong cur, end;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5529
  int mode;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5530
  struct stat64 buf64;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5531
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5532
  if (::fstat64(fd, &buf64) >= 0) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5533
    mode = buf64.st_mode;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5534
    if (S_ISCHR(mode) || S_ISFIFO(mode) || S_ISSOCK(mode)) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5535
      /*
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5536
      * XXX: is the following call interruptible? If so, this might
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5537
      * need to go through the INTERRUPT_IO() wrapper as for other
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5538
      * blocking, interruptible calls in this file.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5539
      */
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5540
      int n,ioctl_return;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5541
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5542
      INTERRUPTIBLE(::ioctl(fd, FIONREAD, &n),ioctl_return,os::Solaris::clear_interrupted);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5543
      if (ioctl_return>= 0) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5544
          *bytes = n;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5545
        return 1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5546
      }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5547
    }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5548
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5549
  if ((cur = ::lseek64(fd, 0L, SEEK_CUR)) == -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5550
    return 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5551
  } else if ((end = ::lseek64(fd, 0L, SEEK_END)) == -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5552
    return 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5553
  } else if (::lseek64(fd, cur, SEEK_SET) == -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5554
    return 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5555
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5556
  *bytes = end - cur;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5557
  return 1;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5558
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5559
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5560
// Map a block of memory.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  5561
char* os::pd_map_memory(int fd, const char* file_name, size_t file_offset,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5562
                     char *addr, size_t bytes, bool read_only,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5563
                     bool allow_exec) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5564
  int prot;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5565
  int flags;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5566
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5567
  if (read_only) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5568
    prot = PROT_READ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5569
    flags = MAP_SHARED;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5570
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5571
    prot = PROT_READ | PROT_WRITE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5572
    flags = MAP_PRIVATE;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5573
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5574
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5575
  if (allow_exec) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5576
    prot |= PROT_EXEC;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5577
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5578
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5579
  if (addr != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5580
    flags |= MAP_FIXED;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5581
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5582
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5583
  char* mapped_address = (char*)mmap(addr, (size_t)bytes, prot, flags,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5584
                                     fd, file_offset);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5585
  if (mapped_address == MAP_FAILED) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5586
    return NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5587
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5588
  return mapped_address;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5589
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5590
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5591
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5592
// Remap a block of memory.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  5593
char* os::pd_remap_memory(int fd, const char* file_name, size_t file_offset,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5594
                       char *addr, size_t bytes, bool read_only,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5595
                       bool allow_exec) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5596
  // same as map_memory() on this OS
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5597
  return os::map_memory(fd, file_name, file_offset, addr, bytes, read_only,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5598
                        allow_exec);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5599
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5600
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5601
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5602
// Unmap a block of memory.
13195
be27e1b6a4b9 6995781: Native Memory Tracking (Phase 1)
zgu
parents: 12735
diff changeset
  5603
bool os::pd_unmap_memory(char* addr, size_t bytes) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5604
  return munmap(addr, bytes) == 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5605
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5606
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5607
void os::pause() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5608
  char filename[MAX_PATH];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5609
  if (PauseAtStartupFile && PauseAtStartupFile[0]) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5610
    jio_snprintf(filename, MAX_PATH, PauseAtStartupFile);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5611
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5612
    jio_snprintf(filename, MAX_PATH, "./vm.paused.%d", current_process_id());
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5613
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5614
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5615
  int fd = ::open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0666);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5616
  if (fd != -1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5617
    struct stat buf;
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5618
    ::close(fd);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5619
    while (::stat(filename, &buf) == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5620
      (void)::poll(NULL, 0, 100);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5621
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5622
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5623
    jio_fprintf(stderr,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5624
      "Could not open pause file '%s', continuing immediately.\n", filename);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5625
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5626
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5627
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5628
#ifndef PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5629
#ifdef INTERPOSE_ON_SYSTEM_SYNCH_FUNCTIONS
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5630
// Turn this on if you need to trace synch operations.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5631
// Set RECORD_SYNCH_LIMIT to a large-enough value,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5632
// and call record_synch_enable and record_synch_disable
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5633
// around the computation of interest.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5634
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5635
void record_synch(char* name, bool returning);  // defined below
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5636
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5637
class RecordSynch {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5638
  char* _name;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5639
 public:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5640
  RecordSynch(char* name) :_name(name)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5641
                 { record_synch(_name, false); }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5642
  ~RecordSynch() { record_synch(_name,   true);  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5643
};
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5644
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5645
#define CHECK_SYNCH_OP(ret, name, params, args, inner)          \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5646
extern "C" ret name params {                                    \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5647
  typedef ret name##_t params;                                  \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5648
  static name##_t* implem = NULL;                               \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5649
  static int callcount = 0;                                     \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5650
  if (implem == NULL) {                                         \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5651
    implem = (name##_t*) dlsym(RTLD_NEXT, #name);               \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5652
    if (implem == NULL)  fatal(dlerror());                      \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5653
  }                                                             \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5654
  ++callcount;                                                  \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5655
  RecordSynch _rs(#name);                                       \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5656
  inner;                                                        \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5657
  return implem args;                                           \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5658
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5659
// in dbx, examine callcounts this way:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5660
// for n in $(eval whereis callcount | awk '{print $2}'); do print $n; done
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5661
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5662
#define CHECK_POINTER_OK(p) \
13728
882756847a04 6964458: Reimplement class meta-data storage to use native memory
coleenp
parents: 13198
diff changeset
  5663
  (!Universe::is_fully_initialized() || !Universe::is_reserved_heap((oop)(p)))
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5664
#define CHECK_MU \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5665
  if (!CHECK_POINTER_OK(mu)) fatal("Mutex must be in C heap only.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5666
#define CHECK_CV \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5667
  if (!CHECK_POINTER_OK(cv)) fatal("Condvar must be in C heap only.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5668
#define CHECK_P(p) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5669
  if (!CHECK_POINTER_OK(p))  fatal(false,  "Pointer must be in C heap only.");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5670
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5671
#define CHECK_MUTEX(mutex_op) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5672
CHECK_SYNCH_OP(int, mutex_op, (mutex_t *mu), (mu), CHECK_MU);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5673
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5674
CHECK_MUTEX(   mutex_lock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5675
CHECK_MUTEX(  _mutex_lock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5676
CHECK_MUTEX( mutex_unlock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5677
CHECK_MUTEX(_mutex_unlock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5678
CHECK_MUTEX( mutex_trylock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5679
CHECK_MUTEX(_mutex_trylock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5680
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5681
#define CHECK_COND(cond_op) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5682
CHECK_SYNCH_OP(int, cond_op, (cond_t *cv, mutex_t *mu), (cv, mu), CHECK_MU;CHECK_CV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5683
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5684
CHECK_COND( cond_wait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5685
CHECK_COND(_cond_wait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5686
CHECK_COND(_cond_wait_cancel);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5687
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5688
#define CHECK_COND2(cond_op) \
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5689
CHECK_SYNCH_OP(int, cond_op, (cond_t *cv, mutex_t *mu, timestruc_t* ts), (cv, mu, ts), CHECK_MU;CHECK_CV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5690
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5691
CHECK_COND2( cond_timedwait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5692
CHECK_COND2(_cond_timedwait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5693
CHECK_COND2(_cond_timedwait_cancel);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5694
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5695
// do the _lwp_* versions too
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5696
#define mutex_t lwp_mutex_t
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5697
#define cond_t  lwp_cond_t
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5698
CHECK_MUTEX(  _lwp_mutex_lock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5699
CHECK_MUTEX(  _lwp_mutex_unlock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5700
CHECK_MUTEX(  _lwp_mutex_trylock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5701
CHECK_MUTEX( __lwp_mutex_lock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5702
CHECK_MUTEX( __lwp_mutex_unlock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5703
CHECK_MUTEX( __lwp_mutex_trylock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5704
CHECK_MUTEX(___lwp_mutex_lock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5705
CHECK_MUTEX(___lwp_mutex_unlock)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5706
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5707
CHECK_COND(  _lwp_cond_wait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5708
CHECK_COND( __lwp_cond_wait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5709
CHECK_COND(___lwp_cond_wait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5710
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5711
CHECK_COND2(  _lwp_cond_timedwait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5712
CHECK_COND2( __lwp_cond_timedwait);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5713
#undef mutex_t
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5714
#undef cond_t
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5715
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5716
CHECK_SYNCH_OP(int, _lwp_suspend2,       (int lwp, int *n), (lwp, n), 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5717
CHECK_SYNCH_OP(int,__lwp_suspend2,       (int lwp, int *n), (lwp, n), 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5718
CHECK_SYNCH_OP(int, _lwp_kill,           (int lwp, int n),  (lwp, n), 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5719
CHECK_SYNCH_OP(int,__lwp_kill,           (int lwp, int n),  (lwp, n), 0);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5720
CHECK_SYNCH_OP(int, _lwp_sema_wait,      (lwp_sema_t* p),   (p),  CHECK_P(p));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5721
CHECK_SYNCH_OP(int,__lwp_sema_wait,      (lwp_sema_t* p),   (p),  CHECK_P(p));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5722
CHECK_SYNCH_OP(int, _lwp_cond_broadcast, (lwp_cond_t* cv),  (cv), CHECK_CV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5723
CHECK_SYNCH_OP(int,__lwp_cond_broadcast, (lwp_cond_t* cv),  (cv), CHECK_CV);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5724
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5725
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5726
// recording machinery:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5727
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5728
enum { RECORD_SYNCH_LIMIT = 200 };
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5729
char* record_synch_name[RECORD_SYNCH_LIMIT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5730
void* record_synch_arg0ptr[RECORD_SYNCH_LIMIT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5731
bool record_synch_returning[RECORD_SYNCH_LIMIT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5732
thread_t record_synch_thread[RECORD_SYNCH_LIMIT];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5733
int record_synch_count = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5734
bool record_synch_enabled = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5735
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5736
// in dbx, examine recorded data this way:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5737
// for n in name arg0ptr returning thread; do print record_synch_$n[0..record_synch_count-1]; done
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5738
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5739
void record_synch(char* name, bool returning) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5740
  if (record_synch_enabled) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5741
    if (record_synch_count < RECORD_SYNCH_LIMIT) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5742
      record_synch_name[record_synch_count] = name;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5743
      record_synch_returning[record_synch_count] = returning;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5744
      record_synch_thread[record_synch_count] = thr_self();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5745
      record_synch_arg0ptr[record_synch_count] = &name;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5746
      record_synch_count++;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5747
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5748
    // put more checking code here:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5749
    // ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5750
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5751
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5752
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5753
void record_synch_enable() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5754
  // start collecting trace data, if not already doing so
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5755
  if (!record_synch_enabled)  record_synch_count = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5756
  record_synch_enabled = true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5757
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5758
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5759
void record_synch_disable() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5760
  // stop collecting trace data
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5761
  record_synch_enabled = false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5762
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5763
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5764
#endif // INTERPOSE_ON_SYSTEM_SYNCH_FUNCTIONS
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5765
#endif // PRODUCT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5766
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5767
const intptr_t thr_time_off  = (intptr_t)(&((prusage_t *)(NULL))->pr_utime);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5768
const intptr_t thr_time_size = (intptr_t)(&((prusage_t *)(NULL))->pr_ttime) -
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5769
                               (intptr_t)(&((prusage_t *)(NULL))->pr_utime);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5770
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5771
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5772
// JVMTI & JVM monitoring and management support
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5773
// The thread_cpu_time() and current_thread_cpu_time() are only
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5774
// supported if is_thread_cpu_time_supported() returns true.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5775
// They are not supported on Solaris T1.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5776
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5777
// current_thread_cpu_time(bool) and thread_cpu_time(Thread*, bool)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5778
// are used by JVM M&M and JVMTI to get user+sys or user CPU time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5779
// of a thread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5780
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5781
// current_thread_cpu_time() and thread_cpu_time(Thread *)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5782
// returns the fast estimate available on the platform.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5783
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5784
// hrtime_t gethrvtime() return value includes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5785
// user time but does not include system time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5786
jlong os::current_thread_cpu_time() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5787
  return (jlong) gethrvtime();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5788
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5789
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5790
jlong os::thread_cpu_time(Thread *thread) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5791
  // return user level CPU time only to be consistent with
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5792
  // what current_thread_cpu_time returns.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5793
  // thread_cpu_time_info() must be changed if this changes
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5794
  return os::thread_cpu_time(thread, false /* user time only */);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5795
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5796
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5797
jlong os::current_thread_cpu_time(bool user_sys_cpu_time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5798
  if (user_sys_cpu_time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5799
    return os::thread_cpu_time(Thread::current(), user_sys_cpu_time);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5800
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5801
    return os::current_thread_cpu_time();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5802
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5803
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5804
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5805
jlong os::thread_cpu_time(Thread *thread, bool user_sys_cpu_time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5806
  char proc_name[64];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5807
  int count;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5808
  prusage_t prusage;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5809
  jlong lwp_time;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5810
  int fd;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5811
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5812
  sprintf(proc_name, "/proc/%d/lwp/%d/lwpusage",
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5813
                     getpid(),
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5814
                     thread->osthread()->lwp_id());
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5815
  fd = ::open(proc_name, O_RDONLY);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5816
  if ( fd == -1 ) return -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5817
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5818
  do {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5819
    count = ::pread(fd,
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5820
                  (void *)&prusage.pr_utime,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5821
                  thr_time_size,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5822
                  thr_time_off);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5823
  } while (count < 0 && errno == EINTR);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  5824
  ::close(fd);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5825
  if ( count < 0 ) return -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5826
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5827
  if (user_sys_cpu_time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5828
    // user + system CPU time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5829
    lwp_time = (((jlong)prusage.pr_stime.tv_sec +
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5830
                 (jlong)prusage.pr_utime.tv_sec) * (jlong)1000000000) +
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5831
                 (jlong)prusage.pr_stime.tv_nsec +
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5832
                 (jlong)prusage.pr_utime.tv_nsec;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5833
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5834
    // user level CPU time only
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5835
    lwp_time = ((jlong)prusage.pr_utime.tv_sec * (jlong)1000000000) +
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5836
                (jlong)prusage.pr_utime.tv_nsec;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5837
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5838
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5839
  return(lwp_time);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5840
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5841
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5842
void os::current_thread_cpu_time_info(jvmtiTimerInfo *info_ptr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5843
  info_ptr->max_value = ALL_64_BITS;      // will not wrap in less than 64 bits
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5844
  info_ptr->may_skip_backward = false;    // elapsed time not wall time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5845
  info_ptr->may_skip_forward = false;     // elapsed time not wall time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5846
  info_ptr->kind = JVMTI_TIMER_USER_CPU;  // only user time is returned
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5847
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5848
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5849
void os::thread_cpu_time_info(jvmtiTimerInfo *info_ptr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5850
  info_ptr->max_value = ALL_64_BITS;      // will not wrap in less than 64 bits
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5851
  info_ptr->may_skip_backward = false;    // elapsed time not wall time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5852
  info_ptr->may_skip_forward = false;     // elapsed time not wall time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5853
  info_ptr->kind = JVMTI_TIMER_USER_CPU;  // only user time is returned
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5854
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5855
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5856
bool os::is_thread_cpu_time_supported() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5857
  if ( os::Solaris::T2_libthread() || UseBoundThreads ) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5858
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5859
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5860
    return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5861
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5862
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5863
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5864
// System loadavg support.  Returns -1 if load average cannot be obtained.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5865
// Return the load average for our processor set if the primitive exists
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5866
// (Solaris 9 and later).  Otherwise just return system wide loadavg.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5867
int os::loadavg(double loadavg[], int nelem) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5868
  if (pset_getloadavg_ptr != NULL) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5869
    return (*pset_getloadavg_ptr)(PS_MYID, loadavg, nelem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5870
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5871
    return ::getloadavg(loadavg, nelem);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5872
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5873
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5874
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5875
//---------------------------------------------------------------------------------
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5876
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5877
bool os::find(address addr, outputStream* st) {
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5878
  Dl_info dlinfo;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5879
  memset(&dlinfo, 0, sizeof(dlinfo));
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5880
  if (dladdr(addr, &dlinfo) != 0) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5881
    st->print(PTR_FORMAT ": ", addr);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5882
    if (dlinfo.dli_sname != NULL && dlinfo.dli_saddr != NULL) {
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5883
      st->print("%s+%#lx", dlinfo.dli_sname, addr-(intptr_t)dlinfo.dli_saddr);
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5884
    } else if (dlinfo.dli_fbase != NULL)
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5885
      st->print("<offset %#lx>", addr-(intptr_t)dlinfo.dli_fbase);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5886
    else
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5887
      st->print("<absolute address>");
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5888
    if (dlinfo.dli_fname != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5889
      st->print(" in %s", dlinfo.dli_fname);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5890
    }
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5891
    if (dlinfo.dli_fbase != NULL) {
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5892
      st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5893
    }
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5894
    st->cr();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5895
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5896
    if (Verbose) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5897
      // decode some bytes around the PC
16670
4af09aff4237 8003310: Enable -Wunused-function when compiling with gcc
mikael
parents: 16669
diff changeset
  5898
      address begin = clamp_address_in_page(addr-40, addr, os::vm_page_size());
4af09aff4237 8003310: Enable -Wunused-function when compiling with gcc
mikael
parents: 16669
diff changeset
  5899
      address end   = clamp_address_in_page(addr+40, addr, os::vm_page_size());
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5900
      address       lowest = (address) dlinfo.dli_sname;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5901
      if (!lowest)  lowest = (address) dlinfo.dli_fbase;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5902
      if (begin < lowest)  begin = lowest;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5903
      Dl_info dlinfo2;
18683
a6418e038255 8015884: runThese crashed with SIGSEGV, hs_err has an error instead of stacktrace
dcubed
parents: 18086
diff changeset
  5904
      if (dladdr(end, &dlinfo2) != 0 && dlinfo2.dli_saddr != dlinfo.dli_saddr
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5905
          && end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5906
        end = (address) dlinfo2.dli_saddr;
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  5907
      Disassembler::decode(begin, end, st);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5908
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5909
    return true;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5910
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5911
  return false;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5912
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5913
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5914
// Following function has been added to support HotSparc's libjvm.so running
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5915
// under Solaris production JDK 1.2.2 / 1.3.0.  These came from
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5916
// src/solaris/hpi/native_threads in the EVM codebase.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5917
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5918
// NOTE: This is no longer needed in the 1.3.1 and 1.4 production release
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5919
// libraries and should thus be removed. We will leave it behind for a while
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5920
// until we no longer want to able to run on top of 1.3.0 Solaris production
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5921
// JDK. See 4341971.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5922
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5923
#define STACK_SLACK 0x800
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5924
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5925
extern "C" {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5926
  intptr_t sysThreadAvailableStackWithSlack() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5927
    stack_t st;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5928
    intptr_t retval, stack_top;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5929
    retval = thr_stksegment(&st);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5930
    assert(retval == 0, "incorrect return value from thr_stksegment");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5931
    assert((address)&st < (address)st.ss_sp, "Invalid stack base returned");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5932
    assert((address)&st > (address)st.ss_sp-st.ss_size, "Invalid stack size returned");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5933
    stack_top=(intptr_t)st.ss_sp-st.ss_size;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5934
    return ((intptr_t)&stack_top - stack_top - STACK_SLACK);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5935
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5936
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5937
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5938
// ObjectMonitor park-unpark infrastructure ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5939
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5940
// We implement Solaris and Linux PlatformEvents with the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5941
// obvious condvar-mutex-flag triple.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5942
// Another alternative that works quite well is pipes:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5943
// Each PlatformEvent consists of a pipe-pair.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5944
// The thread associated with the PlatformEvent
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5945
// calls park(), which reads from the input end of the pipe.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5946
// Unpark() writes into the other end of the pipe.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5947
// The write-side of the pipe must be set NDELAY.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5948
// Unfortunately pipes consume a large # of handles.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5949
// Native solaris lwp_park() and lwp_unpark() work nicely, too.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5950
// Using pipes for the 1st few threads might be workable, however.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5951
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5952
// park() is permitted to return spuriously.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5953
// Callers of park() should wrap the call to park() in
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5954
// an appropriate loop.  A litmus test for the correct
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5955
// usage of park is the following: if park() were modified
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5956
// to immediately return 0 your code should still work,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5957
// albeit degenerating to a spin loop.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5958
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5959
// An interesting optimization for park() is to use a trylock()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5960
// to attempt to acquire the mutex.  If the trylock() fails
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5961
// then we know that a concurrent unpark() operation is in-progress.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5962
// in that case the park() code could simply set _count to 0
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5963
// and return immediately.  The subsequent park() operation *might*
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5964
// return immediately.  That's harmless as the caller of park() is
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5965
// expected to loop.  By using trylock() we will have avoided a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5966
// avoided a context switch caused by contention on the per-thread mutex.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5967
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5968
// TODO-FIXME:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5969
// 1.  Reconcile Doug's JSR166 j.u.c park-unpark with the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5970
//     objectmonitor implementation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5971
// 2.  Collapse the JSR166 parker event, and the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5972
//     objectmonitor ParkEvent into a single "Event" construct.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5973
// 3.  In park() and unpark() add:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5974
//     assert (Thread::current() == AssociatedWith).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5975
// 4.  add spurious wakeup injection on a -XX:EarlyParkReturn=N switch.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5976
//     1-out-of-N park() operations will return immediately.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5977
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5978
// _Event transitions in park()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5979
//   -1 => -1 : illegal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5980
//    1 =>  0 : pass - return immediately
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5981
//    0 => -1 : block
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5982
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5983
// _Event serves as a restricted-range semaphore.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5984
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5985
// Another possible encoding of _Event would be with
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5986
// explicit "PARKED" == 01b and "SIGNALED" == 10b bits.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5987
//
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5988
// TODO-FIXME: add DTRACE probes for:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5989
// 1.   Tx parks
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5990
// 2.   Ty unparks Tx
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5991
// 3.   Tx resumes from park
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5992
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5993
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5994
// value determined through experimentation
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5995
#define ROUNDINGFIX 11
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5996
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5997
// utility to compute the abstime argument to timedwait.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5998
// TODO-FIXME: switch from compute_abstime() to unpackTime().
489c9b5090e2 Initial load
duke
parents:
diff changeset
  5999
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6000
static timestruc_t* compute_abstime(timestruc_t* abstime, jlong millis) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6001
  // millis is the relative timeout time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6002
  // abstime will be the absolute timeout time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6003
  if (millis < 0)  millis = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6004
  struct timeval now;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6005
  int status = gettimeofday(&now, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6006
  assert(status == 0, "gettimeofday");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6007
  jlong seconds = millis / 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6008
  jlong max_wait_period;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6009
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6010
  if (UseLWPSynchronization) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6011
    // forward port of fix for 4275818 (not sleeping long enough)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6012
    // There was a bug in Solaris 6, 7 and pre-patch 5 of 8 where
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6013
    // _lwp_cond_timedwait() used a round_down algorithm rather
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6014
    // than a round_up. For millis less than our roundfactor
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6015
    // it rounded down to 0 which doesn't meet the spec.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6016
    // For millis > roundfactor we may return a bit sooner, but
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6017
    // since we can not accurately identify the patch level and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6018
    // this has already been fixed in Solaris 9 and 8 we will
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6019
    // leave it alone rather than always rounding down.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6020
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6021
    if (millis > 0 && millis < ROUNDINGFIX) millis = ROUNDINGFIX;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6022
       // It appears that when we go directly through Solaris _lwp_cond_timedwait()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6023
           // the acceptable max time threshold is smaller than for libthread on 2.5.1 and 2.6
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6024
           max_wait_period = 21000000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6025
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6026
    max_wait_period = 50000000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6027
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6028
  millis %= 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6029
  if (seconds > max_wait_period) {      // see man cond_timedwait(3T)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6030
     seconds = max_wait_period;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6031
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6032
  abstime->tv_sec = now.tv_sec  + seconds;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6033
  long       usec = now.tv_usec + millis * 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6034
  if (usec >= 1000000) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6035
    abstime->tv_sec += 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6036
    usec -= 1000000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6037
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6038
  abstime->tv_nsec = usec * 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6039
  return abstime;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6040
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6041
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6042
// Test-and-clear _Event, always leaves _Event set to 0, returns immediately.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6043
// Conceptually TryPark() should be equivalent to park(0).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6044
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6045
int os::PlatformEvent::TryPark() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6046
  for (;;) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6047
    const int v = _Event ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6048
    guarantee ((v == 0) || (v == 1), "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6049
    if (Atomic::cmpxchg (0, &_Event, v) == v) return v  ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6050
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6051
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6052
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6053
void os::PlatformEvent::park() {           // AKA: down()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6054
  // Invariant: Only the thread associated with the Event/PlatformEvent
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6055
  // may call park().
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6056
  int v ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6057
  for (;;) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6058
      v = _Event ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6059
      if (Atomic::cmpxchg (v-1, &_Event, v) == v) break ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6060
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6061
  guarantee (v >= 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6062
  if (v == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6063
     // Do this the hard way by blocking ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6064
     // See http://monaco.sfbay/detail.jsf?cr=5094058.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6065
     // TODO-FIXME: for Solaris SPARC set fprs.FEF=0 prior to parking.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6066
     // Only for SPARC >= V8PlusA
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6067
#if defined(__sparc) && defined(COMPILER2)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6068
     if (ClearFPUAtPark) { _mark_fpu_nosave() ; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6069
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6070
     int status = os::Solaris::mutex_lock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6071
     assert_status(status == 0, status,  "mutex_lock");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6072
     guarantee (_nParked == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6073
     ++ _nParked ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6074
     while (_Event < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6075
        // for some reason, under 2.7 lwp_cond_wait() may return ETIME ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6076
        // Treat this the same as if the wait was interrupted
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6077
        // With usr/lib/lwp going to kernel, always handle ETIME
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6078
        status = os::Solaris::cond_wait(_cond, _mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6079
        if (status == ETIME) status = EINTR ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6080
        assert_status(status == 0 || status == EINTR, status, "cond_wait");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6081
     }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6082
     -- _nParked ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6083
     _Event = 0 ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6084
     status = os::Solaris::mutex_unlock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6085
     assert_status(status == 0, status, "mutex_unlock");
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6086
    // Paranoia to ensure our locked and lock-free paths interact
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6087
    // correctly with each other.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6088
    OrderAccess::fence();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6089
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6090
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6091
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6092
int os::PlatformEvent::park(jlong millis) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6093
  guarantee (_nParked == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6094
  int v ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6095
  for (;;) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6096
      v = _Event ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6097
      if (Atomic::cmpxchg (v-1, &_Event, v) == v) break ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6098
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6099
  guarantee (v >= 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6100
  if (v != 0) return OS_OK ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6101
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6102
  int ret = OS_TIMEOUT;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6103
  timestruc_t abst;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6104
  compute_abstime (&abst, millis);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6105
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6106
  // See http://monaco.sfbay/detail.jsf?cr=5094058.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6107
  // For Solaris SPARC set fprs.FEF=0 prior to parking.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6108
  // Only for SPARC >= V8PlusA
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6109
#if defined(__sparc) && defined(COMPILER2)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6110
 if (ClearFPUAtPark) { _mark_fpu_nosave() ; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6111
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6112
  int status = os::Solaris::mutex_lock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6113
  assert_status(status == 0, status, "mutex_lock");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6114
  guarantee (_nParked == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6115
  ++ _nParked ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6116
  while (_Event < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6117
     int status = os::Solaris::cond_timedwait(_cond, _mutex, &abst);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6118
     assert_status(status == 0 || status == EINTR ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6119
                   status == ETIME || status == ETIMEDOUT,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6120
                   status, "cond_timedwait");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6121
     if (!FilterSpuriousWakeups) break ;                // previous semantics
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6122
     if (status == ETIME || status == ETIMEDOUT) break ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6123
     // We consume and ignore EINTR and spurious wakeups.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6124
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6125
  -- _nParked ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6126
  if (_Event >= 0) ret = OS_OK ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6127
  _Event = 0 ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6128
  status = os::Solaris::mutex_unlock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6129
  assert_status(status == 0, status, "mutex_unlock");
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6130
  // Paranoia to ensure our locked and lock-free paths interact
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6131
  // correctly with each other.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6132
  OrderAccess::fence();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6133
  return ret;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6134
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6135
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6136
void os::PlatformEvent::unpark() {
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6137
  // Transitions for _Event:
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6138
  //    0 :=> 1
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6139
  //    1 :=> 1
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6140
  //   -1 :=> either 0 or 1; must signal target thread
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6141
  //          That is, we can safely transition _Event from -1 to either
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6142
  //          0 or 1. Forcing 1 is slightly more efficient for back-to-back
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6143
  //          unpark() calls.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6144
  // See also: "Semaphores in Plan 9" by Mullender & Cox
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6145
  //
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6146
  // Note: Forcing a transition from "-1" to "1" on an unpark() means
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6147
  // that it will take two back-to-back park() calls for the owning
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6148
  // thread to block. This has the benefit of forcing a spurious return
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6149
  // from the first park() call after an unpark() call which will help
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6150
  // shake out uses of park() and unpark() without condition variables.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6151
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6152
  if (Atomic::xchg(1, &_Event) >= 0) return;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6153
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6154
  // If the thread associated with the event was parked, wake it.
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6155
  // Wait for the thread assoc with the PlatformEvent to vacate.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6156
  int status = os::Solaris::mutex_lock(_mutex);
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6157
  assert_status(status == 0, status, "mutex_lock");
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6158
  int AnyWaiters = _nParked;
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6159
  status = os::Solaris::mutex_unlock(_mutex);
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6160
  assert_status(status == 0, status, "mutex_unlock");
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6161
  guarantee(AnyWaiters == 0 || AnyWaiters == 1, "invariant");
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6162
  if (AnyWaiters != 0) {
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6163
    // We intentional signal *after* dropping the lock
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6164
    // to avoid a common class of futile wakeups.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6165
    status = os::Solaris::cond_signal(_cond);
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6166
    assert_status(status == 0, status, "cond_signal");
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6167
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6168
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6169
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6170
// JSR166
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6171
// -------------------------------------------------------
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6172
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6173
/*
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6174
 * The solaris and linux implementations of park/unpark are fairly
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6175
 * conservative for now, but can be improved. They currently use a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6176
 * mutex/condvar pair, plus _counter.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6177
 * Park decrements _counter if > 0, else does a condvar wait.  Unpark
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6178
 * sets count to 1 and signals condvar.  Only one thread ever waits
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6179
 * on the condvar. Contention seen when trying to park implies that someone
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6180
 * is unparking you, so don't wait. And spurious returns are fine, so there
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6181
 * is no need to track notifications.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6182
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6183
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6184
#define MAX_SECS 100000000
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6185
/*
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6186
 * This code is common to linux and solaris and will be moved to a
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6187
 * common place in dolphin.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6188
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6189
 * The passed in time value is either a relative time in nanoseconds
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6190
 * or an absolute time in milliseconds. Either way it has to be unpacked
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6191
 * into suitable seconds and nanoseconds components and stored in the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6192
 * given timespec structure.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6193
 * Given time is a 64-bit value and the time_t used in the timespec is only
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6194
 * a signed-32-bit value (except on 64-bit Linux) we have to watch for
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6195
 * overflow if times way in the future are given. Further on Solaris versions
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6196
 * prior to 10 there is a restriction (see cond_timedwait) that the specified
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6197
 * number of seconds, in abstime, is less than current_time  + 100,000,000.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6198
 * As it will be 28 years before "now + 100000000" will overflow we can
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6199
 * ignore overflow and just impose a hard-limit on seconds using the value
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6200
 * of "now + 100,000,000". This places a limit on the timeout of about 3.17
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6201
 * years from "now".
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6202
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6203
static void unpackTime(timespec* absTime, bool isAbsolute, jlong time) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6204
  assert (time > 0, "convertTime");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6205
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6206
  struct timeval now;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6207
  int status = gettimeofday(&now, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6208
  assert(status == 0, "gettimeofday");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6209
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6210
  time_t max_secs = now.tv_sec + MAX_SECS;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6211
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6212
  if (isAbsolute) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6213
    jlong secs = time / 1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6214
    if (secs > max_secs) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6215
      absTime->tv_sec = max_secs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6216
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6217
    else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6218
      absTime->tv_sec = secs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6219
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6220
    absTime->tv_nsec = (time % 1000) * NANOSECS_PER_MILLISEC;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6221
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6222
  else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6223
    jlong secs = time / NANOSECS_PER_SEC;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6224
    if (secs >= MAX_SECS) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6225
      absTime->tv_sec = max_secs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6226
      absTime->tv_nsec = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6227
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6228
    else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6229
      absTime->tv_sec = now.tv_sec + secs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6230
      absTime->tv_nsec = (time % NANOSECS_PER_SEC) + now.tv_usec*1000;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6231
      if (absTime->tv_nsec >= NANOSECS_PER_SEC) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6232
        absTime->tv_nsec -= NANOSECS_PER_SEC;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6233
        ++absTime->tv_sec; // note: this must be <= max_secs
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6234
      }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6235
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6236
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6237
  assert(absTime->tv_sec >= 0, "tv_sec < 0");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6238
  assert(absTime->tv_sec <= max_secs, "tv_sec > max_secs");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6239
  assert(absTime->tv_nsec >= 0, "tv_nsec < 0");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6240
  assert(absTime->tv_nsec < NANOSECS_PER_SEC, "tv_nsec >= nanos_per_sec");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6241
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6242
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6243
void Parker::park(bool isAbsolute, jlong time) {
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6244
  // Ideally we'd do something useful while spinning, such
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6245
  // as calling unpackTime().
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6246
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6247
  // Optional fast-path check:
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6248
  // Return immediately if a permit is available.
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6249
  // We depend on Atomic::xchg() having full barrier semantics
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6250
  // since we are doing a lock-free update to _counter.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6251
  if (Atomic::xchg(0, &_counter) > 0) return;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6252
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6253
  // Optional fast-exit: Check interrupt before trying to wait
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6254
  Thread* thread = Thread::current();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6255
  assert(thread->is_Java_thread(), "Must be JavaThread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6256
  JavaThread *jt = (JavaThread *)thread;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6257
  if (Thread::is_interrupted(thread, false)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6258
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6259
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6260
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6261
  // First, demultiplex/decode time arguments
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6262
  timespec absTime;
6962
d49132ce025b 6763959: java.util.concurrent.locks.LockSupport.parkUntil(0) blocks forever
acorn
parents: 6176
diff changeset
  6263
  if (time < 0 || (isAbsolute && time == 0) ) { // don't wait at all
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6264
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6265
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6266
  if (time > 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6267
    // Warning: this code might be exposed to the old Solaris time
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6268
    // round-down bugs.  Grep "roundingFix" for details.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6269
    unpackTime(&absTime, isAbsolute, time);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6270
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6271
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6272
  // Enter safepoint region
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6273
  // Beware of deadlocks such as 6317397.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6274
  // The per-thread Parker:: _mutex is a classic leaf-lock.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6275
  // In particular a thread must never block on the Threads_lock while
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6276
  // holding the Parker:: mutex.  If safepoints are pending both the
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6277
  // the ThreadBlockInVM() CTOR and DTOR may grab Threads_lock.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6278
  ThreadBlockInVM tbivm(jt);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6279
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6280
  // Don't wait if cannot get lock since interference arises from
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6281
  // unblocking.  Also. check interrupt before trying wait
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6282
  if (Thread::is_interrupted(thread, false) ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6283
      os::Solaris::mutex_trylock(_mutex) != 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6284
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6285
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6286
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6287
  int status ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6288
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6289
  if (_counter > 0)  { // no wait needed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6290
    _counter = 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6291
    status = os::Solaris::mutex_unlock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6292
    assert (status == 0, "invariant") ;
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6293
    // Paranoia to ensure our locked and lock-free paths interact
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6294
    // correctly with each other and Java-level accesses.
4487
de1359156181 6822370: ReentrantReadWriteLock: threads hung when there are no threads holding onto the lock (Netra x4450)
dholmes
parents: 3594
diff changeset
  6295
    OrderAccess::fence();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6296
    return;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6297
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6298
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6299
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6300
  // Don't catch signals while blocked; let the running threads have the signals.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6301
  // (This allows a debugger to break into the running thread.)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6302
  sigset_t oldsigs;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6303
  sigset_t* allowdebug_blocked = os::Solaris::allowdebug_blocked_signals();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6304
  thr_sigsetmask(SIG_BLOCK, allowdebug_blocked, &oldsigs);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6305
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6306
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6307
  OSThreadWaitState osts(thread->osthread(), false /* not Object.wait() */);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6308
  jt->set_suspend_equivalent();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6309
  // cleared by handle_special_suspend_equivalent_condition() or java_suspend_self()
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6310
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6311
  // Do this the hard way by blocking ...
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6312
  // See http://monaco.sfbay/detail.jsf?cr=5094058.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6313
  // TODO-FIXME: for Solaris SPARC set fprs.FEF=0 prior to parking.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6314
  // Only for SPARC >= V8PlusA
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6315
#if defined(__sparc) && defined(COMPILER2)
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6316
  if (ClearFPUAtPark) { _mark_fpu_nosave() ; }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6317
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6318
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6319
  if (time == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6320
    status = os::Solaris::cond_wait (_cond, _mutex) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6321
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6322
    status = os::Solaris::cond_timedwait (_cond, _mutex, &absTime);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6323
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6324
  // Note that an untimed cond_wait() can sometimes return ETIME on older
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6325
  // versions of the Solaris.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6326
  assert_status(status == 0 || status == EINTR ||
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6327
                status == ETIME || status == ETIMEDOUT,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6328
                status, "cond_timedwait");
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6329
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6330
#ifdef ASSERT
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6331
  thr_sigsetmask(SIG_SETMASK, &oldsigs, NULL);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6332
#endif
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6333
  _counter = 0 ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6334
  status = os::Solaris::mutex_unlock(_mutex);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6335
  assert_status(status == 0, status, "mutex_unlock") ;
15234
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6336
  // Paranoia to ensure our locked and lock-free paths interact
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6337
  // correctly with each other and Java-level accesses.
ff1f01be5fbd 8004902: correctness fixes motivated by contended locking work (6607129)
dcubed
parents: 15096
diff changeset
  6338
  OrderAccess::fence();
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6339
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6340
  // If externally suspended while waiting, re-suspend
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6341
  if (jt->handle_special_suspend_equivalent_condition()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6342
    jt->java_suspend_self();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6343
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6344
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6345
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6346
void Parker::unpark() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6347
  int s, status ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6348
  status = os::Solaris::mutex_lock (_mutex) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6349
  assert (status == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6350
  s = _counter;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6351
  _counter = 1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6352
  status = os::Solaris::mutex_unlock (_mutex) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6353
  assert (status == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6354
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6355
  if (s < 1) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6356
    status = os::Solaris::cond_signal (_cond) ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6357
    assert (status == 0, "invariant") ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6358
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6359
}
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6360
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6361
extern char** environ;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6362
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6363
// Run the specified command in a separate process. Return its exit value,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6364
// or -1 on failure (e.g. can't fork a new process).
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6365
// Unlike system(), this function can be called from signal handler. It
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6366
// doesn't block SIGINT et al.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6367
int os::fork_and_exec(char* cmd) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6368
  char * argv[4];
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6369
  argv[0] = (char *)"sh";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6370
  argv[1] = (char *)"-c";
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6371
  argv[2] = cmd;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6372
  argv[3] = NULL;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6373
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6374
  // fork is async-safe, fork1 is not so can't use in signal handler
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6375
  pid_t pid;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6376
  Thread* t = ThreadLocalStorage::get_thread_slow();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6377
  if (t != NULL && t->is_inside_signal_handler()) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6378
    pid = fork();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6379
  } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6380
    pid = fork1();
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6381
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6382
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6383
  if (pid < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6384
    // fork failed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6385
    warning("fork failed: %s", strerror(errno));
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6386
    return -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6387
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6388
  } else if (pid == 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6389
    // child process
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6390
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6391
    // try to be consistent with system(), which uses "/usr/bin/sh" on Solaris
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6392
    execve("/usr/bin/sh", argv, environ);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6393
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6394
    // execve failed
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6395
    _exit(-1);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6396
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6397
  } else  {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6398
    // copied from J2SE ..._waitForProcessExit() in UNIXProcess_md.c; we don't
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6399
    // care about the actual exit code, for now.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6400
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6401
    int status;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6402
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6403
    // Wait for the child process to exit.  This returns immediately if
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6404
    // the child has already exited. */
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6405
    while (waitpid(pid, &status, 0) < 0) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6406
        switch (errno) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6407
        case ECHILD: return 0;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6408
        case EINTR: break;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6409
        default: return -1;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6410
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6411
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6412
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6413
    if (WIFEXITED(status)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6414
       // The child exited normally; get its exit code.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6415
       return WEXITSTATUS(status);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6416
    } else if (WIFSIGNALED(status)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6417
       // The child exited because of a signal
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6418
       // The best value to return is 0x80 + signal number,
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6419
       // because that is what all Unix shells do, and because
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6420
       // it allows callers to distinguish between process exit and
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6421
       // process death by signal.
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6422
       return 0x80 + WTERMSIG(status);
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6423
    } else {
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6424
       // Unknown exit code; pass it through
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6425
       return status;
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6426
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6427
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
  6428
}
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6429
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6430
// is_headless_jre()
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6431
//
11161
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6432
// Test for the existence of xawt/libmawt.so or libawt_xawt.so
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6433
// in order to report if we are running in a headless jre
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6434
//
11161
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6435
// Since JDK8 xawt/libmawt.so was moved into the same directory
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6436
// as libawt.so, and renamed libawt_xawt.so
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6437
//
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6438
bool os::is_headless_jre() {
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6439
    struct stat statbuf;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6440
    char buf[MAXPATHLEN];
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6441
    char libmawtpath[MAXPATHLEN];
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6442
    const char *xawtstr  = "/xawt/libmawt.so";
11161
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6443
    const char *new_xawtstr = "/libawt_xawt.so";
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6444
    char *p;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6445
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6446
    // Get path to libjvm.so
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6447
    os::jvm_path(buf, sizeof(buf));
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6448
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6449
    // Get rid of libjvm.so
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6450
    p = strrchr(buf, '/');
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6451
    if (p == NULL) return false;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6452
    else *p = '\0';
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6453
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6454
    // Get rid of client or server
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6455
    p = strrchr(buf, '/');
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6456
    if (p == NULL) return false;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6457
    else *p = '\0';
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6458
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6459
    // check xawt/libmawt.so
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6460
    strcpy(libmawtpath, buf);
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6461
    strcat(libmawtpath, xawtstr);
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6462
    if (::stat(libmawtpath, &statbuf) == 0) return false;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6463
11161
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6464
    // check libawt_xawt.so
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6465
    strcpy(libmawtpath, buf);
11161
ec855b5a23c2 7110017: is_headless_jre should be updated to reflect the new location of awt toolkit libraries
dholmes
parents: 10739
diff changeset
  6466
    strcat(libmawtpath, new_xawtstr);
6176
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6467
    if (::stat(libmawtpath, &statbuf) == 0) return false;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6468
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6469
    return true;
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6470
}
4d9030fe341f 6953477: Increase portability and flexibility of building Hotspot
bobv
parents: 5922
diff changeset
  6471
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6472
size_t os::write(int fd, const void *buf, unsigned int nBytes) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6473
  INTERRUPTIBLE_RETURN_INT(::write(fd, buf, nBytes), os::Solaris::clear_interrupted);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6474
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6475
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6476
int os::close(int fd) {
18078
10417cf9967d 7178026: os::close can restart ::close but that is not a restartable syscall
rdurbin
parents: 18069
diff changeset
  6477
  return ::close(fd);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6478
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6479
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6480
int os::socket_close(int fd) {
18078
10417cf9967d 7178026: os::close can restart ::close but that is not a restartable syscall
rdurbin
parents: 18069
diff changeset
  6481
  return ::close(fd);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6482
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6483
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6484
int os::recv(int fd, char* buf, size_t nBytes, uint flags) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6485
  INTERRUPTIBLE_RETURN_INT((int)::recv(fd, buf, nBytes, flags), os::Solaris::clear_interrupted);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6486
}
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6487
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6488
int os::send(int fd, char* buf, size_t nBytes, uint flags) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6489
  INTERRUPTIBLE_RETURN_INT((int)::send(fd, buf, nBytes, flags), os::Solaris::clear_interrupted);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6490
}
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6491
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6492
int os::raw_send(int fd, char* buf, size_t nBytes, uint flags) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6493
  RESTARTABLE_RETURN_INT((int)::send(fd, buf, nBytes, flags));
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6494
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6495
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6496
// As both poll and select can be interrupted by signals, we have to be
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6497
// prepared to restart the system call after updating the timeout, unless
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6498
// a poll() is done with timeout == -1, in which case we repeat with this
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6499
// "wait forever" value.
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6500
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6501
int os::timeout(int fd, long timeout) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6502
  int res;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6503
  struct timeval t;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6504
  julong prevtime, newtime;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6505
  static const char* aNull = 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6506
  struct pollfd pfd;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6507
  pfd.fd = fd;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6508
  pfd.events = POLLIN;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6509
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6510
  gettimeofday(&t, &aNull);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6511
  prevtime = ((julong)t.tv_sec * 1000)  +  t.tv_usec / 1000;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6512
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6513
  for(;;) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6514
    INTERRUPTIBLE_NORESTART(::poll(&pfd, 1, timeout), res, os::Solaris::clear_interrupted);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6515
    if(res == OS_ERR && errno == EINTR) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6516
        if(timeout != -1) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6517
          gettimeofday(&t, &aNull);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6518
          newtime = ((julong)t.tv_sec * 1000)  +  t.tv_usec /1000;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6519
          timeout -= newtime - prevtime;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6520
          if(timeout <= 0)
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6521
            return OS_OK;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6522
          prevtime = newtime;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6523
        }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6524
    } else return res;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6525
  }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6526
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6527
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6528
int os::connect(int fd, struct sockaddr *him, socklen_t len) {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6529
  int _result;
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6530
  INTERRUPTIBLE_NORESTART(::connect(fd, him, len), _result,\
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6531
                          os::Solaris::clear_interrupted);
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6532
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6533
  // Depending on when thread interruption is reset, _result could be
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6534
  // one of two values when errno == EINTR
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6535
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6536
  if (((_result == OS_INTRPT) || (_result == OS_ERR))
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6537
      && (errno == EINTR)) {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6538
     /* restarting a connect() changes its errno semantics */
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6539
     INTERRUPTIBLE(::connect(fd, him, len), _result,\
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6540
                   os::Solaris::clear_interrupted);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6541
     /* undo these changes */
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6542
     if (_result == OS_ERR) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6543
       if (errno == EALREADY) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6544
         errno = EINPROGRESS; /* fall through */
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6545
       } else if (errno == EISCONN) {
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6546
         errno = 0;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6547
         return OS_OK;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6548
       }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6549
     }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6550
   }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6551
   return _result;
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6552
 }
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6553
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6554
int os::accept(int fd, struct sockaddr* him, socklen_t* len) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6555
  if (fd < 0) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6556
    return OS_ERR;
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6557
  }
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6558
  INTERRUPTIBLE_RETURN_INT((int)::accept(fd, him, len),\
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6559
                           os::Solaris::clear_interrupted);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6560
}
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6561
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6562
int os::recvfrom(int fd, char* buf, size_t nBytes, uint flags,
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6563
                 sockaddr* from, socklen_t* fromlen) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6564
  INTERRUPTIBLE_RETURN_INT((int)::recvfrom(fd, buf, nBytes, flags, from, fromlen),\
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6565
                           os::Solaris::clear_interrupted);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6566
}
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6567
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6568
int os::sendto(int fd, char* buf, size_t len, uint flags,
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6569
               struct sockaddr* to, socklen_t tolen) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6570
  INTERRUPTIBLE_RETURN_INT((int)::sendto(fd, buf, len, flags, to, tolen),\
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6571
                           os::Solaris::clear_interrupted);
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6572
}
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6573
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6574
int os::socket_available(int fd, jint *pbytes) {
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6575
  if (fd < 0) {
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6576
    return OS_OK;
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6577
  }
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6578
  int ret;
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6579
  RESTARTABLE(::ioctl(fd, FIONREAD, pbytes), ret);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6580
  // note: ioctl can return 0 when successful, JVM_SocketAvailable
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6581
  // is expected to return 0 on failure and 1 on success to the jdk.
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6582
  return (ret == OS_ERR) ? 0 : 1;
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6583
}
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6584
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6585
int os::bind(int fd, struct sockaddr* him, socklen_t len) {
7405
e6fc8d3926f8 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 7397
diff changeset
  6586
   INTERRUPTIBLE_RETURN_INT_NORESTART(::bind(fd, him, len),\
11256
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6587
                                      os::Solaris::clear_interrupted);
025cd1741566 7091417: recvfrom's 6th input should be of type socklen_t
phh
parents: 11161
diff changeset
  6588
}
13198
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6589
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6590
// Get the default path to the core file
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6591
// Returns the length of the string
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6592
int os::get_core_path(char* buffer, size_t bufferSize) {
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6593
  const char* p = get_current_directory(buffer, bufferSize);
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6594
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6595
  if (p == NULL) {
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6596
    assert(p != NULL, "failed to get current directory");
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6597
    return 0;
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6598
  }
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6599
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6600
  return strlen(buffer);
271c557a7623 7129724: MAC: Core file location is wrong in crash report
mikael
parents: 13195
diff changeset
  6601
}
19546
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6602
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6603
#ifndef PRODUCT
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6604
void TestReserveMemorySpecial_test() {
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6605
  // No tests available for this platform
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6606
}
f6b7c9e96ea3 8007074: SIGSEGV at ParMarkBitMap::verify_clear()
stefank
parents: 18703
diff changeset
  6607
#endif