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

com.ibm.as400.access.RemoteCommandImplNative Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename:  RemoteCommandImplNative.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 1999-2007 International Business Machines Corporation and
// others.  All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
//
// @A1 - 9/18/2007 - Changes to follow proxy command chain.
//
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.io.IOException;


// The RemoteCommandImplNative class is the native implementation of CommandCall and ProgramCall.
class RemoteCommandImplNative extends RemoteCommandImplRemote
{
  private static final String CLASSNAME = "com.ibm.as400.access.RemoteCommandImplNative";
  static
  {
    if (Trace.traceOn_) Trace.logLoadPath(CLASSNAME);
  }

    static
    {
 	   NativeMethods.loadNativeLibraryQyjspart(); 
    }

    // Report whether the RemoteCommandImpl object is a native object.
    public boolean isNative()
    {
      return true;
    }

    // Connects to the server.
    // @param threadSafety  The assumed thread safety of the command/program.
    protected void open(Boolean threadSafety) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      // Note:  LOOKUP_THREADSAFETY is null so == must be used
      if ((LOOKUP_THREADSAFETY == threadSafety) || OFF_THREAD.equals(threadSafety)) {
        openOffThread();
      }
      else {
        openOnThread();
      }
    }

    protected void openOnThread() throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native implementation object open.");

      // If converter was not set with a user override ccsid, set converter to job ccsid.
      if (!ccsidIsUserOveride_ && (converter_ == null))
      {
        converter_ = ConverterImplRemote.getConverter(system_.getCcsid(), system_);
      }
      if (AS400.nativeVRM.vrm_ >= 0x00050300)
      {
        if (AS400.nativeVRM.vrm_ >= 0x00060100)
        {
          serverDataStreamLevel_ = 10;
          if (unicodeConverter_ == null) {
            unicodeConverter_ = ConverterImplRemote.getConverter(1200, system_);
          }
        }
        else
        {
          serverDataStreamLevel_ = 7;
        }
      }

      // Set the secondary language library on the server.
      if (system_.isMustAddLanguageLibrary() &&
          !system_.isSkipFurtherSettingOfLanguageLibrary()) // see if we should try
      {
        // Note: If we were going through the Remote Command Host Server, the host server would set the secondary language library for us.
        // Since we're not using the host server, we need to handle this ourselves.
        // We need to do this on every open, since several different threads may be using this RemoteCommandImpl object.

        // Retrieve the name of the secondary language library (if any).
        String secLibName = retrieveSecondaryLanguageLibName();  // never returns null
        // Set the NLV on server to match the client's locale.
        if (secLibName.length() != 0)
        {
          setNlvOnServer(secLibName);
        }
        // Retain result, to avoid repeated library lookups for same system object.
        system_.setLanguageLibrary(secLibName);
        // Set to non-null, to indicate we already looked-up the value.
      }

    }


    // Retrieves the secondary language library (if any).
    // If fail to retrieve library name, or name is blank, returns "".
    private String retrieveSecondaryLanguageLibName()
    {
      String secLibName = system_.getLanguageLibrary();
      if (secLibName == null)  // 'null' implies not already looked-up
      {
        String clientNLV = system_.getNLV(); // NLV of client (based on locale)
        try
        {
          int ccsid = system_.getCcsid();
          ConvTable conv = ConvTable.getTable(ccsid, null);

          ProgramParameter[] parameterList = new ProgramParameter[6];
          int len = 108+10; // length of PRDR0100, plus first 10 bytes of PRDR0200
          parameterList[0] = new ProgramParameter(len); // receiver variable - PRDR0100 plus first 10 bytes of PRDR0200
          parameterList[1] = new ProgramParameter(BinaryConverter.intToByteArray(len)); // length of receiver variable
          parameterList[2] = new ProgramParameter(conv.stringToByteArray("PRDR0200")); // format name

          byte[] productInfo = new byte[36];  // product information
          AS400Text text4 = new AS400Text(4, ccsid, system_);
          AS400Text text6 = new AS400Text(6, ccsid, system_);
          AS400Text text7 = new AS400Text(7, ccsid, system_);
          AS400Text text10 = new AS400Text(10, ccsid, system_);
          text7.toBytes("*OPSYS", productInfo, 0);  // product ID
          text6.toBytes("*CUR", productInfo, 7);  // release level
          text4.toBytes("0000", productInfo, 13);  // product option
          text10.toBytes(clientNLV, productInfo, 17); // load ID (specifies desired NLV)
          BinaryConverter.intToByteArray(36, productInfo, 28);  // length of product information parm
          BinaryConverter.intToByteArray(ccsid, productInfo, 32);  // ccsid for returned directory
          parameterList[3] = new ProgramParameter(productInfo); // product information
          parameterList[4] = new ProgramParameter(new byte[4]); // error code
          parameterList[5] = new ProgramParameter(conv.stringToByteArray("PRDI0200")); // product information format name

          // Call QSZRTVPR (Retrieve Product Information) to retrieve the library for the secondary language.
          // Note: QSZRTVPR is documented as non-threadsafe. However, the API owner has indicated that this API will never alter the state of the system, and that it cannot damage the system; so it can safely be called on-thread.
          boolean succeeded = runProgramOnThread("QSYS", "QSZRTVPR", parameterList, AS400Message.MESSAGE_OPTION_UP_TO_10, true);
          // Note: This method is only called from within open().
          // The final parm indicates that the on-thread open() has already been done (on this thread).

          if (!succeeded)
          {
            Trace.log(Trace.WARNING, "Unable to retrieve secondary language library name for NLV " + clientNLV, new AS400Exception(messageList_));
          }
          else
          {
            byte[] outputData = parameterList[0].getOutputData();
            int offsetToAddlInfo = BinaryConverter.byteArrayToInt(outputData, 84);
            secLibName = conv.byteArrayToString(outputData, offsetToAddlInfo, 10).trim();
            if (secLibName.length() == 0) {
              Trace.log(Trace.WARNING, "Unable to retrieve secondary language library name for NLV " + clientNLV + ": Blank library name returned.");
            }
          }
        }
        catch (Throwable t) {
          Trace.log(Trace.WARNING, "Unable to retrieve secondary language library name for NLV " + clientNLV, t);
        }
      }

      return (secLibName == null ? "" : secLibName);
    }

    // Sets the NLV (for the current thread) on the server, so that system msgs are returned in correct language.
    private void setNlvOnServer(String secondaryLibraryName)
    {
      if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native implementation object setting national language for messages.");
      try
      {
        // Call CHGSYSLIBL (Change System Library List) to add the library for the secondary language.
        // Note: According to the spec, CHGSYSLIBL "changes the system portion of the library list for the current thread".
        // Prior to V6R1, CHGSYSLIBL is documented as non-threadsafe.  However, the CL owner has indicated that this CL has actually been threadsafe all along, and that it cannot damage the system; so it can safely be called on-thread.
        // At worst, if system value QMLTTHDACN == 3, the system will simply refuse to execute the command.  In which case, the secondary language library won't get added.
        String cmd = "QSYS/CHGSYSLIBL LIB("+secondaryLibraryName+") OPTION(*ADD)";
        boolean succeeded = runCommandOnThread(cmd, AS400Message.MESSAGE_OPTION_UP_TO_10, true);
        // Note: This method is only called from within open().
        // The final parm indicates that the on-thread open() has already been done (on this thread).

        if (!succeeded)
        {
          if (messageList_.length !=0)
          {
            if (messageList_[0].getID().equals("CPF2103")) // lib is already in list
            {
              // Tolerate this error.  It means that we're good to go.
              // If this is the very first native open() for this system_, set flag to indicate that the lib is already in list by default.  This will eliminate clutter in the job log, from subsequent attempts to set it.
              if (system_.getLanguageLibrary() == null) { // null implies first native open
                system_.setSkipFurtherSettingOfLanguageLibrary(); // don't keep trying on subsequent open's
              }
            }
            else if (messageList_[0].getID().equals("CPD0032")) // not auth'd to call CHGSYSLIBL
            {
              system_.setSkipFurtherSettingOfLanguageLibrary(); // don't keep trying on subsequent open's
              Trace.log(Trace.DIAGNOSTIC, "Profile " + system_.getUserId() + " not authorized to use CHGSYSLIBL to add secondary language library " + secondaryLibraryName + " to liblist.");
              // Note: The Remote Command Host Server runs this command under greater authority.
            }
            else if (messageList_[0].getID().equals("CPF2110")) // library not found
            {
              system_.setSkipFurtherSettingOfLanguageLibrary();  // don't keep trying on subsequent open's
              Trace.log(Trace.WARNING, "Secondary language library " + secondaryLibraryName + " was not found.");
            }
            else
            {
              Trace.log(Trace.ERROR, "Unable to add secondary language library " + secondaryLibraryName + " to library list.", new AS400Exception(messageList_));
            }
          }
          else  // no system messages returned
          {
            Trace.log(Trace.WARNING, "Unable to add secondary language library " + secondaryLibraryName + " to library list.");
          }
        }
      }
      catch (Throwable t)
      {
        Trace.log(Trace.WARNING, "Failed to add secondary language library " + secondaryLibraryName + " to library list.", t);
      }
    }

    //For native and thread safe, it gets the jvm job info. Otherwise, get the job info of host server.
    public String getJobInfo(Boolean threadSafety) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {

      if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting job information from implementation object.");

      // Note: The runProgram() method that we call below, will call the appropriate open() method.

      // Set up the parameter list for the program that we will use to get the job information (QWCRTVCA).
      ProgramParameter[] parameterList = new ProgramParameter[6];

      // First parameter:  receiver variable - output - char(*).
      // RTVC0100's 20-byte header, plus 26 bytes for "job name" field.
      byte[] dataReceived = new byte[46];
      parameterList[0] = new ProgramParameter(dataReceived.length);

      // Second parameter:  length of receiver variable - input - binary(4).
      byte[] receiverLength = BinaryConverter.intToByteArray(dataReceived.length);
      parameterList[1] = new ProgramParameter(receiverLength);

      // Third parameter:  format name - input - char(8).
      // Set to EBCDIC "RTVC0100".
      byte[] formatName = {(byte)0xD9, (byte)0xE3, (byte)0xE5, (byte)0xC3, (byte)0xF0, (byte)0xF1, (byte)0xF0, (byte)0xF0};
      parameterList[2] = new ProgramParameter(formatName);

      // Fourth parameter:  number of attributes to return - input - binary(4).
      byte[] numAttributes = BinaryConverter.intToByteArray(1); // 1 attribute.
      parameterList[3] = new ProgramParameter(numAttributes);

      // Fifth parameter:  key of attributes to be returned - input - array(*) of binary(4).
      byte[] attributeKey = BinaryConverter.intToByteArray(1009); // "Job name."
      parameterList[4] = new ProgramParameter(attributeKey);

      // Sixth parameter:  error code - input/output - char(*).
      // Eight bytes of zero's indicates to throw exceptions.
      // Send as input because we are not interested in the output.
      parameterList[5] = new ProgramParameter(new byte[8]);

      // Prepare to call the "Retrieve Current Attributes" API.
      // Design note: QWCRTVCA is documented to be conditionally threadsafe.

      // Note: Depending upon whether the program represented by this ProgramCall object will be run on-thread or through the host servers (as indicated by the 'threadsafety' flag), we will issue the job info query accordingly, either on-thread or through the host servers, in order to get the appropriate Job.

      // Retrieve Current Attributes.  Failure is returned as a message list.
      try
      {
          boolean succeeded;
          if (ON_THREAD.equals(threadSafety)) {
            succeeded = runProgramOnThread("QSYS", "QWCRTVCA", parameterList, MESSAGE_OPTION_DEFAULT, false);
          }
          else {
            succeeded = runProgramOffThread("QSYS", "QWCRTVCA", parameterList, MESSAGE_OPTION_DEFAULT);
          }
          if (!succeeded)
          {
              Trace.log(Trace.ERROR, "Unable to retrieve job information.");
              throw new AS400Exception(messageList_);
          }
      }
      catch (ObjectDoesNotExistException e)
      {
          Trace.log(Trace.ERROR, "Unexpected ObjectDoesNotExistException:", e);
          throw new InternalErrorException(InternalErrorException.UNEXPECTED_EXCEPTION);
      }

      // Get the data returned from the program.
      dataReceived = parameterList[0].getOutputData();
      if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Job information retrieved:", dataReceived);

      // Examine the "job name" field.  26 bytes starting at offset 20.  The format of the job name is a 10-character simple job name, a 10-character user name, and a 6-character job number.
      return converter_.byteArrayToString(dataReceived, 20, 26);
    }

    // This method is reserved for use by other Impl classes in this package.
    public boolean runCommand(String command) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      // The caller didn't specify whether to call the command on- or off-thread.
      // Base the decision on the setting the "threadSafe" system property.

      if (shouldRunOnThread(command)) {
        return runCommandOnThread(command, MESSAGE_OPTION_DEFAULT, false);
      }
      else {
        return runCommandOffThread(command, MESSAGE_OPTION_DEFAULT);
      }
    }


    // Runs the command.
    // @param threadSafety  The assumed thread safety of the command/program.
    // @return  true if command is successful; false otherwise.
    public boolean runCommand(String command, Boolean threadSafety, int messageOption) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      boolean runOnThread;

      if (ON_THREAD.equals(threadSafety)) {
        runOnThread = true;
      }
      else if (OFF_THREAD.equals(threadSafety)) {
        runOnThread = false;
      }
      else // threadSafety == LOOKUP_THREADSAFETY
      {
        // Look up the command's indicated threadsafety on the system.
        if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "LOOKING-UP thread safety of command: " + command);
        runOnThread = (getThreadsafeIndicator(command) == THREADSAFE_INDICATED_YES);
      }

      if (runOnThread) {
        return runCommandOnThread(command, MESSAGE_OPTION_DEFAULT, false);
      }
      else {
        if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Delegating runCommand() to super class.");
        return runCommandOffThread(command, messageOption);
      }
    }

    // Runs the command.
    // @return  true if command is successful; false otherwise.
    private boolean runCommandOnThread(String command, int messageOption, boolean currentlyOpeningOnThisThread) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
        if (Trace.traceOn_)
        {
          Trace.log(Trace.INFORMATION, "Native implementation running command: " + command);
          Trace.log(Trace.DIAGNOSTIC, "Running command ON-THREAD: " + command);
        }
        if (!currentlyOpeningOnThisThread) openOnThread();

        if (AS400.nativeVRM.vrm_ >= 0x00060100)
        {
            return runCommandOnThread(unicodeConverter_.stringToByteArray(command), messageOption, 1200);
        }
        return runCommandOnThread(converter_.stringToByteArray(command), messageOption, 0);
    }

    public boolean runCommand(byte[] commandAsBytes, String commandAsString) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      // The caller didn't specify whether to call the command on- or off-thread.
      // Base the decision on the setting the "threadSafe" system property.

      if (shouldRunOnThread(commandAsString)) {
        openOnThread();
        if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Running command ON-THREAD: " + commandAsString);
        return runCommandOnThread(commandAsBytes, MESSAGE_OPTION_DEFAULT, 0);
      }
      else {
        if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Running command OFF-THREAD: " + commandAsString);
        return runCommandOffThread(commandAsBytes, MESSAGE_OPTION_DEFAULT, 0);
      }
    }

    private boolean runCommandOnThread(byte[] commandBytes, int messageOption, int ccsid) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
        byte[] swapToPH = new byte[12];
        byte[] swapFromPH = new byte[12];
        boolean didSwap = system_.swapTo(swapToPH, swapFromPH);
        if (OFF_THREAD.equals(priorCallWasOnThread_))
        {
          if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Prior call was off-thread, but this call is on-thread, so different job.");
        }
        priorCallWasOnThread_ = ON_THREAD;

        try
        {
            if (Trace.traceOn_) Trace.log(Trace.INFORMATION, "Invoking native method.");
            if (AS400.nativeVRM.vrm_ < 0x00050300)
            {
                try
                {
                    byte[] replyBytes = runCommandNative(commandBytes);
                    if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native reply bytes:", replyBytes);

                    if (replyBytes == null) replyBytes = new byte[0];

                    // Get info from reply.
                    messageList_ = RemoteCommandImplNative.parseMessages(replyBytes, converter_);
                    return true;
                }
                catch (NativeException e)  // Exception found by C code.
                {
                    messageList_ = RemoteCommandImplNative.parseMessages(e.data, converter_);
                    return false;
                }
            }
            else
            {
                try
                {
                    byte[] replyBytes = AS400.nativeVRM.vrm_ < 0x00060100 ? runCommandNativeV5R3(commandBytes, messageOption) : NativeMethods.runCommand(commandBytes, ccsid, messageOption);
                    if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native reply bytes:", replyBytes);

                    // Get info from reply.
                    messageList_ = RemoteCommandImplNative.parseMessagesV5R3(replyBytes, converter_);
                    return true;
                }
                catch (NativeException e)  // Exception found by C code.
                {
                    messageList_ = RemoteCommandImplNative.parseMessagesV5R3(e.data, converter_);
                    return false;
                }
            }
        }
        finally
        {
            if (didSwap) system_.swapBack(swapToPH, swapFromPH);
        }
    }

    public boolean runProgram(String library, String name, ProgramParameter[] parameterList) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException
    {
      // The caller didn't specify whether to call the command on- or off-thread.
      // Base the decision on the setting the "threadSafe" system property.

      String property = ProgramCall.getThreadSafetyProperty();
      Boolean threadSafety;
      if (property != null && property.equals("true")) {
        threadSafety = ON_THREAD;
      }
      else threadSafety = OFF_THREAD;

      return runProgram(library, name, parameterList, threadSafety);
    }

    public boolean runProgram(String library, String name, ProgramParameter[] parameterList, Boolean threadSafety) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException
    {
      // Note: We don't have a way to look up the thread safety of programs.
      if (ON_THREAD.equals(threadSafety)) {
        return runProgramOnThread(library, name, parameterList, MESSAGE_OPTION_DEFAULT, false);
      }
      else {
        return runProgramOffThread(library, name, parameterList, MESSAGE_OPTION_DEFAULT);
      }
    }

    public boolean runProgram(String library, String name, ProgramParameter[] parameterList, Boolean threadSafety, int messageOption) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException
    {
      // Note: We don't have a way to look up the thread safety of programs.
      if (ON_THREAD.equals(threadSafety)) {
        return runProgramOnThread(library, name, parameterList, messageOption, false);
      }
      else {
        return runProgramOffThread(library, name, parameterList, messageOption);
      }
    }

    // Run the program.
    protected boolean runProgramOnThread(String library, String name, ProgramParameter[] parameterList, int messageOption, boolean currentlyOpeningOnThisThread) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException
    {
        if (Trace.traceOn_)
        {
          Trace.log(Trace.INFORMATION, "Native implementation running program: " + library + "/" + name);
          Trace.log(Trace.DIAGNOSTIC, "Running program ON-THREAD: " + library + "/" + name);
        }
        if (OFF_THREAD.equals(priorCallWasOnThread_))
        {
          if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Prior call was off-thread, but this call is on-thread, so different job.");
        }
        priorCallWasOnThread_ = ON_THREAD;

        // Run the program on-thread.
        if (!currentlyOpeningOnThisThread) openOnThread();

        if (AS400.nativeVRM.vrm_ < 0x00050300)
        {
            // Create a "call program" request, and write it as raw bytes to a byte array.
            // Set up the buffer that contains the program to call.  The buffer contains three items:
            //  10 characters - the program to call.
            //  10 characters - the library that contains the program.
            //   4 bytes      - the number of parameters.
            byte[] programNameBuffer = {(byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x40, (byte)0x00, (byte)0x00, (byte)0x00, (byte)0x00};
            converter_.stringToByteArray(name, programNameBuffer);
            converter_.stringToByteArray(library, programNameBuffer, 10);
            BinaryConverter.intToByteArray(parameterList.length, programNameBuffer, 20);

            // Set up the parameter structure.  There is one structure for each parameters.
            // The structure contains:
            //   4 bytes - the length of the parameter.
            //   2 bytes - the parameters usage (input/output/inout).
            //   4 bytes - the offset into the parameter buffer.
            byte[] programParameterStructure = new byte[parameterList.length * 10];
            int totalParameterLength = 0;
            for (int i = 0, offset = 0; i < parameterList.length; ++i)
            {
                int parameterMaxLength = parameterList[i].getMaxLength();
                int parameterUsage = parameterList[i].getUsage();
                if (parameterUsage == ProgramParameter.NULL)
                {
                    // Server does not allow null parameters.
                    parameterUsage = ProgramParameter.INPUT;
                }
                BinaryConverter.intToByteArray(parameterMaxLength, programParameterStructure, i * 10);
                BinaryConverter.unsignedShortToByteArray(parameterUsage, programParameterStructure, i * 10 + 4);
                BinaryConverter.intToByteArray(offset, programParameterStructure, i * 10 + 6);

                offset += parameterMaxLength;
                totalParameterLength += parameterMaxLength;
            }

            // Set up the Parameter area.
            byte[] programParameters = new byte[totalParameterLength];
            for (int i = 0, offset = 0; i < parameterList.length; ++i)
            {
                byte[] inputData = parameterList[i].getInputData();
                int parameterMaxLength = parameterList[i].getMaxLength();
                if (inputData != null)
                {
                    System.arraycopy(inputData, 0, programParameters, offset, inputData.length);
                }
                offset += parameterMaxLength;
            }

            if (Trace.traceOn_)
            {
                Trace.log(Trace.DIAGNOSTIC, "Program name bytes:", programNameBuffer);
                Trace.log(Trace.DIAGNOSTIC, "Program parameter bytes:", programParameterStructure);
                Trace.log(Trace.DIAGNOSTIC, "Program parameters:", programParameters);
            }
            byte[] swapToPH = new byte[12];
            byte[] swapFromPH = new byte[12];
            boolean didSwap = system_.swapTo(swapToPH, swapFromPH);
            try
            {
                // Call native method.
                if (Trace.traceOn_) Trace.log(Trace.INFORMATION, "Invoking native method.");
                byte[] replyBytes = runProgramNative(programNameBuffer, programParameterStructure, programParameters);
                if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native reply bytes:", replyBytes);

                // Reset the message list.
                messageList_ = new AS400Message[0];

                // For each output/inout parm, in order, set data returned.
                for (int index = 0, i = 0; i < parameterList.length; ++i)
                {
                    int parameterMaxLength = parameterList[i].getMaxLength();
                    int outputDataLength = parameterList[i].getOutputDataLength();
                    if (outputDataLength > 0)
                    {
                        byte[] outputData = new byte[outputDataLength];
                        System.arraycopy(replyBytes, index, outputData, 0, outputDataLength);
                        parameterList[i].setOutputData(outputData);
                    }
                    index += parameterMaxLength;
                }
                return true;
            }
            catch (NativeException e)  // Exception found by C code.
            {
                messageList_ = RemoteCommandImplNative.parseMessages(e.data, converter_);
                if (messageList_.length == 0) return false;

                // Parse information from byte array.
                String id = messageList_[messageList_.length - 1].getID();

                if (id.equals("MCH3401"))
                {
                    byte[] substitutionBytes = messageList_[messageList_.length - 1].getSubstitutionData();
                    if (substitutionBytes[0] == 0x02 && substitutionBytes[1] == 0x01 && name.equals(converter_.byteArrayToString(substitutionBytes, 2, 30).trim()))
                    {
                        throw new ObjectDoesNotExistException(QSYSObjectPathName.toPath(library, name, "PGM"), ObjectDoesNotExistException.OBJECT_DOES_NOT_EXIST);
                    }
                    if (substitutionBytes[0] == 0x04 && substitutionBytes[1] == 0x01 && library.equals(converter_.byteArrayToString(substitutionBytes, 2, 30).trim()))
                    {
                        throw new ObjectDoesNotExistException(QSYSObjectPathName.toPath(library, name, "PGM"), ObjectDoesNotExistException.LIBRARY_DOES_NOT_EXIST);
                    }
                }
                return false;
            }
            finally
            {
                if (didSwap) system_.swapBack(swapToPH, swapFromPH);
            }
        }
        else
        {
            byte[] tempBytes = converter_.stringToByteArray(name);
            byte[] nameBytes = new byte[tempBytes.length + 1];
            System.arraycopy(tempBytes, 0, nameBytes, 0, tempBytes.length);
            tempBytes = converter_.stringToByteArray(library);
            byte[] libraryBytes = new byte[tempBytes.length + 1];
            System.arraycopy(tempBytes, 0, libraryBytes, 0, tempBytes.length);

            byte[] offsetArray = new byte[parameterList.length * 4];
            int totalParameterLength = 0;
            for (int i = 0; i < parameterList.length; ++i)
            {
                if (parameterList[i].getUsage() == ProgramParameter.NULL)
                {
                    BinaryConverter.intToByteArray(-1, offsetArray, i * 4);
                }
                else
                {
                    BinaryConverter.intToByteArray(totalParameterLength, offsetArray, i * 4);
                }
                totalParameterLength += parameterList[i].getMaxLength();
            }

            // Set up the Parameter area.
            byte[] programParameters = new byte[totalParameterLength];
            for (int i = 0, offset = 0; i < parameterList.length; ++i)
            {
                byte[] inputData = parameterList[i].getInputData();
                if (inputData != null)
                {
                    System.arraycopy(inputData, 0, programParameters, offset, inputData.length);
                }
                offset += parameterList[i].getMaxLength();
            }

            if (Trace.traceOn_)
            {
                Trace.log(Trace.DIAGNOSTIC, "Program name bytes:", nameBytes);
                Trace.log(Trace.DIAGNOSTIC, "Program library bytes:", libraryBytes);
                Trace.log(Trace.DIAGNOSTIC, "Number of parameters:", parameterList.length);
                Trace.log(Trace.DIAGNOSTIC, "Offset array:", offsetArray);
                Trace.log(Trace.DIAGNOSTIC, "Program parameters:", programParameters);
            }
            byte[] swapToPH = new byte[12];
            byte[] swapFromPH = new byte[12];
            boolean didSwap = system_.swapTo(swapToPH, swapFromPH);
            try
            {
                // Call native method.
                if (Trace.traceOn_) Trace.log(Trace.INFORMATION, "Invoking native method.");
                byte[] replyBytes = AS400.nativeVRM.vrm_ < 0x00060100 ? runProgramNativeV5R3(nameBytes, libraryBytes, parameterList.length, offsetArray, programParameters, messageOption) : NativeMethods.runProgram(nameBytes, libraryBytes, parameterList.length, offsetArray, programParameters, messageOption);
                if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Native reply bytes:", replyBytes);

                // Reset the message list.
                messageList_ = new AS400Message[0];

                // For each output/inout parm, in order, set data returned.
                for (int index = 0, i = 0; i < parameterList.length; ++i)
                {
                    int outputDataLength = parameterList[i].getOutputDataLength();
                    if (outputDataLength > 0)
                    {
                        byte[] outputData = new byte[outputDataLength];
                        System.arraycopy(replyBytes, index, outputData, 0, outputDataLength);
                        parameterList[i].setOutputData(outputData);
                    }
                    index += parameterList[i].getMaxLength();
                }
                return true;
            }
            catch (NativeException e)  // Exception found by C code.
            {
                messageList_ = RemoteCommandImplNative.parseMessagesV5R3(e.data, converter_);
                if (messageList_.length == 0) return false;

                // Parse information from byte array.
                String id = messageList_[messageList_.length - 1].getID();

                if (id.equals("MCH3401"))
                {
                    byte[] substitutionBytes = messageList_[messageList_.length - 1].getSubstitutionData();
                    if (substitutionBytes[0] == 0x02 && substitutionBytes[1] == 0x01 && name.equals(converter_.byteArrayToString(substitutionBytes, 2, 30).trim()))
                    {
                        throw new ObjectDoesNotExistException(QSYSObjectPathName.toPath(library, name, "PGM"), ObjectDoesNotExistException.OBJECT_DOES_NOT_EXIST);
                    }
                    if (substitutionBytes[0] == 0x04 && substitutionBytes[1] == 0x01 && library.equals(converter_.byteArrayToString(substitutionBytes, 2, 30).trim()))
                    {
                        throw new ObjectDoesNotExistException(QSYSObjectPathName.toPath(library, name, "PGM"), ObjectDoesNotExistException.LIBRARY_DOES_NOT_EXIST);
                    }
                }
                return false;
            }
            finally
            {
                if (didSwap) system_.swapBack(swapToPH, swapFromPH);
            }
        }
    }

    static AS400Message[] parseMessages(byte[] data, ConverterImplRemote converter)
    {
        int messageNumber = data.length / 10240;
        AS400Message[] messageList = new AS400Message[messageNumber];

        for (int offset = 0, i = 0; i < messageNumber; ++i)
        {
            messageList[i] = new AS400Message();
            messageList[i].setID(converter.byteArrayToString(data, offset + 12, 7));
            messageList[i].setType((data[offset + 19] & 0x0F) * 10 + (data[offset + 20] & 0x0F));
            messageList[i].setSeverity(BinaryConverter.byteArrayToInt(data, offset + 8));
            messageList[i].setFileName(converter.byteArrayToString(data, offset + 25, 10).trim());
            messageList[i].setLibraryName(converter.byteArrayToString(data, offset + 45, 10).trim());

            int substitutionDataLength = BinaryConverter.byteArrayToInt(data, offset + 80);
            int textLength = BinaryConverter.byteArrayToInt(data, offset + 88);

            byte[] substitutionData = new byte[substitutionDataLength];
            System.arraycopy(data, offset + 112, substitutionData, 0, substitutionDataLength);
            messageList[i].setSubstitutionData(substitutionData);

            messageList[i].setText(converter.byteArrayToString(data, offset + 112 + substitutionDataLength, textLength));
            offset += 10240;
        }
        return messageList;
    }

    static AS400Message[] parseMessagesV5R3(byte[] data, ConverterImplRemote converter)
    {
        int messageNumber = BinaryConverter.byteArrayToInt(data, 0);
        AS400Message[] messageList = new AS400Message[messageNumber];

        for (int offset = 4, i = 0; i < messageNumber; ++i)
        {
            messageList[i] = new AS400Message();
            messageList[i].setID(converter.byteArrayToString(data, offset + 12, 7));
            messageList[i].setType((data[offset + 19] & 0x0F) * 10 + (data[offset + 20] & 0x0F));
            messageList[i].setSeverity(BinaryConverter.byteArrayToInt(data, offset + 8));
            messageList[i].setFileName(converter.byteArrayToString(data, offset + 25, 10).trim());
            messageList[i].setLibraryName(converter.byteArrayToString(data, offset + 45, 10).trim());

            int substitutionDataLength = BinaryConverter.byteArrayToInt(data, offset + 80);
            int textLength = BinaryConverter.byteArrayToInt(data, offset + 88);
            int helpLength = BinaryConverter.byteArrayToInt(data, offset + 96);

            byte[] substitutionData = new byte[substitutionDataLength];
            System.arraycopy(data, offset + 112, substitutionData, 0, substitutionDataLength);
            messageList[i].setSubstitutionData(substitutionData);

            messageList[i].setText(converter.byteArrayToString(data, offset + 112 + substitutionDataLength, textLength));
            messageList[i].setHelp(converter.byteArrayToString(data, offset + 112 + substitutionDataLength + textLength, helpLength));

            offset += BinaryConverter.byteArrayToInt(data, offset);
            offset += BinaryConverter.byteArrayToInt(data, offset);
        }
        return messageList;
    }


    // Determines whether or not the command should be called on-thread.
    private final boolean shouldRunOnThread(String command) throws AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException
    {
      boolean runOnThread;

      // Check the threadSafe property, and apply it if set.
      String property = CommandCall.getThreadSafetyProperty();

      if ((property == null) || (property.equals("false"))) {
        runOnThread = false;
      }
      else if (property.equals("true")) {
        runOnThread = true;
      }
      else if (property.equals("lookup")) {
        // Look it up on the system.
        if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "LOOKING-UP thread safety of command: " + command);
        runOnThread = (getThreadsafeIndicator(command) == THREADSAFE_INDICATED_YES);
      }
      else {
        runOnThread = false;
        // Assume the utility method has logged a warning about unrecognized property value.
      }
      return runOnThread;
    }

    private native byte[] runCommandNative(byte[] command) throws NativeException;
    private static native byte[] runCommandNativeV5R3(byte[] command, int messageOption) throws NativeException;
    private native byte[] runProgramNative(byte[] programNameBuffer, byte[] programParameterStructure, byte[] programParameters) throws NativeException;
    private static native byte[] runProgramNativeV5R3(byte[] name, byte[] library, int numberParameters, byte[] offsetArray, byte[] programParameters, int messageOption) throws NativeException;
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy