All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.spf4j.unix.JVMArguments Maven / Gradle / Ivy

Go to download

A continuously growing collection of utilities to measure performance, get better diagnostics, improve performance, or do things more reliably, faster that other open source libraries...

There is a newer version: 8.10.0
Show newest version
/*
 * Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 *
 * Additionally licensed with:
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.spf4j.unix;

import com.google.common.io.Files;
import com.sun.jna.StringArray;
import com.sun.jna.Memory;
import com.sun.jna.Native;
import static com.sun.jna.Pointer.NULL;
import com.sun.jna.ptr.IntByReference;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;

import java.io.IOException;
import java.io.File;
import java.io.ByteArrayOutputStream;
import java.io.RandomAccessFile;
import java.io.DataInputStream;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.Objects;
import java.util.function.Function;
import javax.annotation.Nullable;
import org.spf4j.base.Runtime;
import org.spf4j.unix.CLibrary.FILE;

/**
 * List of arguments for Java VM and application. based on class from akuma lib (http://akuma.kohsuke.org).
 */
@SuppressFBWarnings("DM_DEFAULT_ENCODING")
// default char encoding is used on purpose.
public final class JVMArguments {

  private final List arguments;

  public JVMArguments(final int size) {
    arguments = new ArrayList<>(size);
  }

  public JVMArguments(final Collection c) {
    arguments = new ArrayList<>(c);
  }

  public String getExecutable() {
    return arguments.get(0);
  }

  /**
   * Removes the first System property.
   * @param pname the name of the system property to remove.
   * @return the value of the removed system property. or null if there is no such property.
   */
  @Nullable
  public String removeSystemProperty(final String pname) {
    Iterator itr = arguments.iterator();
    itr.next(); // skip command.
    int nl = pname.length();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith("-D") && s.regionMatches(2, pname, 0, nl)) {
        int l = nl + 2;
        if (s.length() == l) {
          itr.remove();
          return "";
        } else if (s.charAt(l) == '=') {
          itr.remove();
          return s.substring(l + 1);
        }
      }
    }
    return null;
  }

  @Nullable
  public String getSystemProperty(final String pname) {
    Iterator itr = arguments.iterator();
    itr.next(); // skip command.
    int nl = pname.length();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith("-D") && s.regionMatches(2, pname, 0, nl)) {
        int l = nl + 2;
        if (s.length() == l) {
          return "";
        } else if (s.charAt(l) == '=') {
          return s.substring(l + 1);
        }
      }
    }
    return null;
  }

  @Nullable
  public void createOrUpdateSystemProperty(final String pname, final Function replacer) {
    ListIterator itr = arguments.listIterator();
    itr.next(); // skip command.
    int nl = pname.length();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith("-D") && s.regionMatches(2, pname, 0, nl)) {
        int l = nl + 2;
        if (s.length() == l) {
          itr.set("-D" + pname + '=' + replacer.apply(""));
          return;
        } else if (s.charAt(l) == '=') {
          itr.set("-D" + pname + '=' + replacer.apply(s.substring(l + 1)));
          return;
        }
      }
    }
    arguments.add(1, "-D" + pname + '=' + replacer.apply(null));
  }


  public boolean hasSystemProperty(final String pname) {
    Iterator itr = arguments.iterator();
    itr.next(); // skip command.
    int nl = pname.length();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith("-D") && s.regionMatches(2, pname, 0, nl)) {
        int l = nl + 2;
        if (s.length() == l) {
          return true;
        } else if (s.charAt(l) == '=') {
          return true;
        }
      }
    }
    return false;
  }
  /**
   * remove all system properties starting with a prefix.
   * @param pname the prefix
   * @return number of system properties removed.
   */
  public int removeAllSystemPropertiesStartingWith(final String pname) {
    String name = "-D" + pname;
    int nrRemoved = 0;
    Iterator itr = arguments.iterator();
    itr.next();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith(name)) {
        itr.remove();
        nrRemoved++;
      }
    }
    return nrRemoved;
  }

  public void setSystemProperty(final String name, final String value) {
    removeSystemProperty(name);
    // index 0 is the executable name
    arguments.add(1, "-D" + name + '=' + value);
  }

  public void setVMArgument(final String argument) {
    if (!hasVMArgument(argument)) {
      arguments.add(1, argument);
    }
  }

  public boolean removeVMArgument(final String argument) {
    Iterator itr = arguments.iterator();
    itr.next();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.equals(argument)) {
        itr.remove();
        return true;
      }
    }
    return false;
  }

  public boolean removeVMArgumentStartingWith(final String argument) {
    Iterator itr = arguments.iterator();
    itr.next();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith(argument)) {
        itr.remove();
        return true;
      }
    }
    return false;
  }

  public boolean hasVMArgument(final String argument) {
    Iterator itr = arguments.iterator();
    itr.next();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.equals(argument)) {
        return true;
      }
    }
    return false;
  }

  public boolean hasVMArgumentStartingWith(final String argumentPrefix) {
    Iterator itr = arguments.iterator();
    itr.next();
    while (itr.hasNext()) {
      String s = itr.next();
      if (s.startsWith(argumentPrefix)) {
        return true;
      }
    }
    return false;
  }


  public void add(final String arg) {
    arguments.add(arg);
  }

  /**
   * Removes the n items from the end. Useful for removing all the Java arguments to rebuild them.
   */
  public void removeTail(final int n) {
    int size = arguments.size();
    arguments.removeAll(arguments.subList(size - n, size));
  }

  public StringArray toStringArray() {
    return new StringArray(arguments.toArray(new String[arguments.size()]));
  }

  /**
   * Gets the process argument list of the current process.
   */
  public static JVMArguments current() throws IOException {
    return of(-1);
  }

  /**
   * Gets the process argument list of the specified process ID.
   *
   * @param pid -1 to indicate the current process.
   */
  public static JVMArguments of(final int pid) throws IOException {
    String os = Runtime.OS_NAME;
    switch (os) {
      case "Linux":
        return ofLinux(pid);
      case "SunOS":
        return ofSolaris(pid);
      case "Mac OS X":
        return ofMac(pid);
      case "FreeBSD":
        return ofFreeBSD(pid);
      default:
        throw new UnsupportedOperationException("Unsupported Operating System " + os);
    }
  }

  private static JVMArguments ofLinux(final int ppid) throws IOException {
    int pid = resolvePID(ppid);
    String cmdline = Files.asCharSource(new File("/proc/" + pid + "/cmdline"), Charset.defaultCharset()).read();
    return new JVMArguments(java.util.Arrays.asList(cmdline.split("\0")));
  }

  private static int resolvePID(final int pid) {
    if (pid == -1) {
      return Runtime.PID;
    } else {
      return pid;
    }
  }

  private static JVMArguments ofSolaris(final int ppid) throws IOException {
    // /proc shows different contents based on the caller's memory model, so we need to know if we are 32 or 64.
    // 32 JVMs are the norm, so err on the 32bit side.
    boolean areWe64 = "64".equals(System.getProperty("sun.arch.data.model"));
    int pid = resolvePID(ppid);
    try (RandomAccessFile psinfo = new RandomAccessFile(new File("/proc/" + pid + "/psinfo"), "r")) {
      // see http://cvs.opensolaris.org/source/xref/onnv/onnv-gate/usr/src/uts/common/sys/procfs.h
      //typedef struct psinfo {
      // int pr_flag; /* process flags */
      // int pr_nlwp; /* number of lwps in the process */
      // pid_t pr_pid; /* process id */
      // pid_t pr_ppid; /* process id of parent */
      // pid_t pr_pgid; /* process id of process group leader */
      // pid_t pr_sid; /* session id */
      // uid_t pr_uid; /* real user id */
      // uid_t pr_euid; /* effective user id */
      // gid_t pr_gid; /* real group id */
      // gid_t pr_egid; /* effective group id */
      // uintptr_t pr_addr; /* address of process */
      // size_t pr_size; /* size of process image in Kbytes */
      // size_t pr_rssize; /* resident set size in Kbytes */
      // dev_t pr_ttydev; /* controlling tty device (or PRNODEV) */
      // ushort_t pr_pctcpu; /* % of recent cpu time used by all lwps */
      // ushort_t pr_pctmem; /* % of system memory used by process */
      // timestruc_t pr_start; /* process start time, from the epoch */
      // timestruc_t pr_time; /* cpu time for this process */
      // timestruc_t pr_ctime; /* cpu time for reaped children */
      // char pr_fname[PRFNSZ]; /* name of exec'ed file */
      // char pr_psargs[PRARGSZ]; /* initial characters of arg list */
      // int pr_wstat; /* if zombie, the wait() status */
      // int pr_argc; /* initial argument count */
      // uintptr_t pr_argv; /* address of initial argument vector */
      // uintptr_t pr_envp; /* address of initial environment vector */
      // char pr_dmodel; /* data model of the process */
      // lwpsinfo_t pr_lwp; /* information for representative lwp */
      //} psinfo_t;

      // see http://cvs.opensolaris.org/source/xref/onnv/onnv-gate/usr/src/uts/common/sys/types.h
      // for the size of the various datatype.
      // see http://cvs.opensolaris.org/source/xref/onnv/onnv-gate/usr/src/cmd/ptools/pargs/pargs.c
      // for how to read this information
      psinfo.seek(8);
      if (adjust(psinfo.readInt()) != pid) {
        throw new IOException("psinfo PID mismatch: " + pid + ", " + psinfo);   // sanity check
      }
      /* The following program computes the offset:
                    #include 
                    #include 
                    int main() {
                      printf("psinfo_t = %d\n", sizeof(psinfo_t));
                      psinfo_t *x;
                      x = 0;
                      printf("%x\n", &(x->pr_argc));
                    }
       */

      psinfo.seek(areWe64 ? 0xEC : 0xBC);  // now jump to pr_argc
      int argc = adjust(psinfo.readInt());
      long argp = areWe64 ? adjust(psinfo.readLong()) : to64(adjust(psinfo.readInt()));
      File asFile = new File("/proc/" + pid + "/as");
      if (areWe64) {
        // 32bit and 64bit basically does the same thing, but because the stream position
        // is computed with signed long, doing 64bit seek to a position bigger than Long.MAX_VALUE
        // requres some real hacking. Hence two different code path.
        // (RandomAccessFile uses Java long for offset, so it just can't get to anywhere beyond Long.MAX_VALUE)
        FILE fp = CLibrary.INSTANCE.fopen(asFile.getPath(), "r");
        try {
          JVMArguments args = new JVMArguments(16);
          Memory m = new Memory(8);
          for (int n = 0; n < argc; n++) {
            // read a pointer to one entry
            seek64(fp, argp + ((long) n) * 8);
            m.setLong(0, 0); // just to make sure failed read won't result in bogus value
            CLibrary.INSTANCE.fread(m, 1, 8, fp);
            long p = m.getLong(0);
            args.add(readLine(fp, p));
          }
          return args;
        } finally {
          CLibrary.INSTANCE.fclose(fp);
        }
      } else {
        try (RandomAccessFile as = new RandomAccessFile(asFile, "r")) {
          JVMArguments args = new JVMArguments(16);
          for (int n = 0; n < argc; n++) {
            // read a pointer to one entry
            as.seek(argp + n * 4);
            int p = adjust(as.readInt());

            args.add(readLine(as, p));
          }
          return args;
        }
      }
    }
  }

  /**
   * Seek to the specified position. This method handles offset bigger than {@link Long#MAX_VALUE} correctly.
   *
   * @param upos This value is interpreted as unsigned 64bit integer (even though it's typed 'long')
   */
  private static void seek64(final FILE fp, final long pupos) {
    long upos = pupos;
    CLibrary.INSTANCE.fseek(fp, 0, 0); // start at the beginning
    while (upos < 0) {
      long chunk = Long.MAX_VALUE;
      upos -= chunk;
      CLibrary.INSTANCE.fseek(fp, chunk, 1);
    }
    CLibrary.INSTANCE.fseek(fp, upos, 1);
  }

  /**
   * {@link DataInputStream} reads a value in big-endian, so convert it to the correct value on little-endian systems.
   */
  private static int adjust(final int i) {
    if (Runtime.IS_LITTLE_ENDIAN) {
      return (i << 24) | ((i << 8) & 0x00FF0000) | ((i >> 8) & 0x0000FF00) | (i >>> 24);
    } else {
      return i;
    }
  }

  private static long adjust(final long i) {
    if (Runtime.IS_LITTLE_ENDIAN) {
      return (i << 56)
              | ((i << 40) & 0x00FF000000000000L)
              | ((i << 24) & 0x0000FF0000000000L)
              | ((i << 8) & 0x000000FF00000000L)
              | ((i >> 8) & 0x00000000FF000000L)
              | ((i >> 24) & 0x0000000000FF0000L)
              | ((i >> 40) & 0x000000000000FF00L)
              | (i >> 56);
    } else {
      return i;
    }
  }

  /**
   * int to long conversion with zero-padding.
   */
  private static long to64(final int i) {
    return i & 0xFFFFFFFFL;
  }

  private static String readLine(final RandomAccessFile as, final int p) throws IOException {
    as.seek(to64(p));
    ByteArrayOutputStream buf = new ByteArrayOutputStream();
    int ch;
    while ((ch = as.read()) > 0) {
      buf.write(ch);
    }
    return buf.toString();
  }

  private static String readLine(final FILE as, final long p) {
    seek64(as, p);
    Memory m = new Memory(1);
    ByteArrayOutputStream buf = new ByteArrayOutputStream();
    while (true) {
      if (CLibrary.INSTANCE.fread(m, 1, 1, as) == 0) {
        break;
      }
      byte b = m.getByte(0);
      if (b == 0) {
        break;
      }
      buf.write(b);
    }
    return buf.toString();
  }

  /**
   * Mac support
   *
   * See http://developer.apple.com/qa/qa2001/qa1123.html http://www.osxfaq.com/man/3/kvm_getprocs.ws
   * http://matburt.net/?p=16 (libkvm is removed from OSX) where is kinfo_proc?
   * http://lists.apple.com/archives/xcode-users/2008/Mar/msg00781.html
   *
   * This code uses sysctl to get the arg/env list:
   * http://www.psychofx.com/psi/trac/browser/psi/trunk/src/arch/macosx/macosx_process.c which came from
   * http://www.opensource.apple.com/darwinsource/10.4.2/top-15/libtop.c
   *
   * sysctl is defined in libc.
   *
   * PS source code for Mac: http://www.opensource.apple.com/darwinsource/10.4.1/adv_cmds-79.1/ps.tproj/
   */
  @SuppressFBWarnings("PRMC_POSSIBLY_REDUNDANT_METHOD_CALLS")
  private static JVMArguments ofMac(final int pid) {
    // local constants
    final int ctlKern = 1;
    final int kernArgMax = 8;
    final int kernProcArgs2 = 49;
    final int sizeOfInt = Native.getNativeSize(int.class);
    IntByReference ibf = new IntByReference();

    IntByReference argmaxRef = new IntByReference(0);
    IntByReference size = new IntByReference(sizeOfInt);

    // for some reason, I was never able to get sysctlbyname work.
//        if(LIBC.sysctlbyname("kern.argmax", argmaxRef.getPointer(), size, NULL, ibf)!=0)
    if (CLibrary.INSTANCE.sysctl(new int[]{ctlKern, kernArgMax}, 2, argmaxRef.getPointer(), size, NULL, ibf) != 0) {
      throw new UnsupportedOperationException("Failed to get kernl.argmax: "
              + CLibrary.INSTANCE.strerror(Native.getLastError()));
    }

    int argmax = argmaxRef.getValue();
    StringArrayMemory m = new StringArrayMemory(argmax, sizeOfInt);
    size.setValue(argmax);
    if (CLibrary.INSTANCE.sysctl(new int[]{ctlKern, kernProcArgs2, resolvePID(pid)}, 3, m, size, NULL, ibf) != 0) {
      throw new UnsupportedOperationException("Failed to obtain ken.procargs2: "
              + CLibrary.INSTANCE.strerror(Native.getLastError()));
    }
    /*
         * Make a sysctl() call to get the raw argument space of the
         * process.  The layout is documented in start.s, which is part
         * of the Csu project.  In summary, it looks like:
         *
         * /---------------\ 0x00000000
         * :               :
         * :               :
         * |---------------|
         * | argc          |
         * |---------------|
         * | arg[0]        |
         * |---------------|
         * :               :
         * :               :
         * |---------------|
         * | arg[argc - 1] |
         * |---------------|
         * | 0             |
         * |---------------|
         * | env[0]        |
         * |---------------|
         * :               :
         * :               :
         * |---------------|
         * | env[n]        |
         * |---------------|
         * | 0             |
         * |---------------| <-- Beginning of data returned by sysctl()
         * | exec_path     |     is here.
         * |:::::::::::::::|
         * |               |
         * | String area.  |
         * |               |
         * |---------------| <-- Top of stack.
         * :               :
         * :               :
         * \---------------/ 0xffffffff
     */
    int nargs = m.readInt();
    JVMArguments args = new JVMArguments(nargs);
    m.readString(); // exec path
    for (int i = 0; i < nargs; i++) {
      m.skip0();
      args.add(m.readString());
    }
    return args;
  }

  @SuppressFBWarnings("EQ_DOESNT_OVERRIDE_EQUALS")
  private static final class StringArrayMemory extends Memory {

    private long offset = 0;
    private final int sizeOfInt;

    StringArrayMemory(final long l, final int sizeOfInt) {
      super(l);
      this.sizeOfInt = sizeOfInt;
    }

    private int readInt() {
      int r = getInt(offset);
      offset += sizeOfInt;
      return r;
    }

    private String readString() {
      ByteArrayOutputStream baos = new ByteArrayOutputStream();
      byte ch;
      while ((ch = getByte(offset++)) != '\0') {
        baos.write(ch);
      }
      return baos.toString();
    }

    void skip0() {
      // skip trailing '\0's
      while (getByte(offset) == '\0') {
        offset++;
      }
    }
  }

  private static JVMArguments ofFreeBSD(final int pid) {
    // taken from sys/sysctl.h
    final int ctlKern = 1;
    final int kernArgMax = 8;
    final int kernProc = 14;
    final int kernProcArgs = 7;

    IntByReference ibr = new IntByReference();
    IntByReference sysctlArgMax = new IntByReference();
    IntByReference size = new IntByReference();

    size.setValue(4);
    if (CLibrary.INSTANCE.sysctl(new int[]{ctlKern, kernArgMax},
            2, sysctlArgMax.getPointer(), size, NULL, ibr) != 0) {
      throw new UnsupportedOperationException("Failed to sysctl kern.argmax");
    }

    int argmax = sysctlArgMax.getValue();
    Memory m = new Memory(argmax);
    size.setValue(argmax);

    if (CLibrary.INSTANCE.sysctl(new int[]{ctlKern, kernProc, kernProcArgs, resolvePID(pid)},
            4, m, size, NULL, ibr) != 0) {
      throw new UnsupportedOperationException();
    }

    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    ArrayList lArgs = new ArrayList();
    byte ch;
    int offset = 0;
    while (offset < size.getValue()) {
      while ((ch = m.getByte(offset++)) != '\0') {
        baos.write(ch);
      }
      lArgs.add(baos.toString());
      baos.reset();
    }

    return new JVMArguments(lArgs);
  }

  @Override
  public String toString() {
    return "JVMArguments{" + "arguments=" + arguments + '}';
  }

  @Override
  public int hashCode() {
    return 11 * 7 + Objects.hashCode(this.arguments);
  }

  @Override
  public boolean equals(final Object obj) {
    if (this == obj) {
      return true;
    }
    if (obj == null) {
      return false;
    }
    if (getClass() != obj.getClass()) {
      return false;
    }
    final JVMArguments other = (JVMArguments) obj;
    return Objects.equals(this.arguments, other.arguments);
  }

  public String[] toArray() {
    return arguments.toArray(new String[arguments.size()]);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy