Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Qiu Shumin <shumin.qiu@intel.com> Reviewed-by: Jaben Carsey <jaben.carsey@intel.com> git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17583 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			460 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			460 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file
 | 
						|
  Main file for Parse shell level 2 function.
 | 
						|
 | 
						|
  (C) Copyright 2013-2015 Hewlett-Packard Development Company, L.P.<BR>
 | 
						|
  Copyright (c) 2009 - 2012, Intel Corporation. All rights reserved.<BR>
 | 
						|
  This program and the accompanying materials
 | 
						|
  are licensed and made available under the terms and conditions of the BSD License
 | 
						|
  which accompanies this distribution.  The full text of the license may be found at
 | 
						|
  http://opensource.org/licenses/bsd-license.php
 | 
						|
 | 
						|
  THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | 
						|
  WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | 
						|
 | 
						|
**/
 | 
						|
 | 
						|
#include "UefiShellLevel2CommandsLib.h"
 | 
						|
 | 
						|
/**
 | 
						|
  Check if data is coming from StdIn output.
 | 
						|
 | 
						|
  @param[in] None
 | 
						|
 
 | 
						|
  @retval TRUE  StdIn stream data available to parse 
 | 
						|
  @retval FALSE StdIn stream data is not available to parse. 
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
IsStdInDataAvailable (
 | 
						|
  VOID
 | 
						|
  )
 | 
						|
{
 | 
						|
  SHELL_FILE_HANDLE FileHandle;
 | 
						|
  EFI_STATUS        Status;
 | 
						|
  CHAR16            CharBuffer; 
 | 
						|
  UINTN             CharSize;
 | 
						|
  UINT64            OriginalFilePosition;
 | 
						|
 | 
						|
  Status               = EFI_SUCCESS; 
 | 
						|
  FileHandle           = NULL;
 | 
						|
  OriginalFilePosition = 0;
 | 
						|
 | 
						|
  if (ShellOpenFileByName (L">i", &FileHandle, EFI_FILE_MODE_READ, 0) == EFI_SUCCESS) {
 | 
						|
    CharSize = sizeof(CHAR16);
 | 
						|
    gEfiShellProtocol->GetFilePosition (FileHandle, &OriginalFilePosition);
 | 
						|
    Status = gEfiShellProtocol->ReadFile (FileHandle, &CharSize, &CharBuffer);
 | 
						|
    if (EFI_ERROR (Status) || (CharSize != sizeof(CHAR16))) {
 | 
						|
      return FALSE;
 | 
						|
    }
 | 
						|
    gEfiShellProtocol->SetFilePosition(FileHandle, OriginalFilePosition);
 | 
						|
  }
 | 
						|
 | 
						|
  if (FileHandle == NULL) {
 | 
						|
    return FALSE;
 | 
						|
  } else {
 | 
						|
    return TRUE;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Function to read a single line (up to but not including the \n) using StdIn data from a SHELL_FILE_HANDLE.
 | 
						|
 | 
						|
  If the position upon start is 0, then the Ascii Boolean will be set.  This should be
 | 
						|
  maintained and not changed for all operations with the same file.
 | 
						|
 | 
						|
  @param[in]       Handle        SHELL_FILE_HANDLE to read from.
 | 
						|
  @param[in, out]  Buffer        The pointer to buffer to read into.
 | 
						|
  @param[in, out]  Size          The pointer to number of bytes in Buffer.
 | 
						|
  @param[in]       Truncate      If the buffer is large enough, this has no effect.
 | 
						|
                                 If the buffer is is too small and Truncate is TRUE,
 | 
						|
                                 the line will be truncated.
 | 
						|
                                 If the buffer is is too small and Truncate is FALSE,
 | 
						|
                                 then no read will occur.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS           The operation was successful.  The line is stored in
 | 
						|
                                Buffer.
 | 
						|
  @retval EFI_INVALID_PARAMETER Handle was NULL.
 | 
						|
  @retval EFI_INVALID_PARAMETER Size was NULL.
 | 
						|
  @retval EFI_BUFFER_TOO_SMALL  Size was not large enough to store the line.
 | 
						|
                                Size was updated to the minimum space required.
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
ShellFileHandleReadStdInLine(
 | 
						|
  IN SHELL_FILE_HANDLE          Handle,
 | 
						|
  IN OUT CHAR16                 *Buffer,
 | 
						|
  IN OUT UINTN                  *Size,
 | 
						|
  IN BOOLEAN                    Truncate
 | 
						|
  )
 | 
						|
{
 | 
						|
  EFI_STATUS  Status;
 | 
						|
  CHAR16      CharBuffer;
 | 
						|
  UINTN       CharSize;
 | 
						|
  UINTN       CountSoFar;
 | 
						|
  UINT64      OriginalFilePosition;
 | 
						|
 | 
						|
 | 
						|
  if (Handle == NULL
 | 
						|
    ||Size   == NULL
 | 
						|
   ){
 | 
						|
    return (EFI_INVALID_PARAMETER);
 | 
						|
  }
 | 
						|
  if (Buffer == NULL) {
 | 
						|
    ASSERT(*Size == 0);
 | 
						|
  } else {
 | 
						|
    *Buffer = CHAR_NULL;
 | 
						|
  }
 | 
						|
  gEfiShellProtocol->GetFilePosition (Handle, &OriginalFilePosition);
 | 
						|
 | 
						|
  for (CountSoFar = 0;;CountSoFar++){
 | 
						|
    CharBuffer = 0;
 | 
						|
    CharSize = sizeof(CHAR16);
 | 
						|
    Status = gEfiShellProtocol->ReadFile (Handle, &CharSize, &CharBuffer);
 | 
						|
    if (  EFI_ERROR(Status)
 | 
						|
       || CharSize == 0
 | 
						|
       || (CharBuffer == L'\n')
 | 
						|
     ){
 | 
						|
      break;
 | 
						|
    }
 | 
						|
    //
 | 
						|
    // if we have space save it...
 | 
						|
    //
 | 
						|
    if ((CountSoFar+1)*sizeof(CHAR16) < *Size){
 | 
						|
      ASSERT(Buffer != NULL);
 | 
						|
      ((CHAR16*)Buffer)[CountSoFar] = CharBuffer;
 | 
						|
      ((CHAR16*)Buffer)[CountSoFar+1] = CHAR_NULL;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // if we ran out of space tell when...
 | 
						|
  //
 | 
						|
  if ((CountSoFar+1)*sizeof(CHAR16) > *Size){
 | 
						|
    *Size = (CountSoFar+1)*sizeof(CHAR16);
 | 
						|
    if (!Truncate) {
 | 
						|
      gEfiShellProtocol->SetFilePosition(Handle, OriginalFilePosition);
 | 
						|
    } else {
 | 
						|
      DEBUG((DEBUG_WARN, "The line was truncated in ShellFileHandleReadLine"));
 | 
						|
    }
 | 
						|
    return (EFI_BUFFER_TOO_SMALL);
 | 
						|
  }
 | 
						|
  while(Buffer[StrLen(Buffer)-1] == L'\r') {
 | 
						|
    Buffer[StrLen(Buffer)-1] = CHAR_NULL;
 | 
						|
  }
 | 
						|
 | 
						|
  return (Status);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Function to read a single line using StdIn from a SHELL_FILE_HANDLE. The \n is not included in the returned
 | 
						|
  buffer.  The returned buffer must be callee freed.
 | 
						|
 | 
						|
  If the position upon start is 0, then the Ascii Boolean will be set.  This should be
 | 
						|
  maintained and not changed for all operations with the same file.
 | 
						|
 | 
						|
  @param[in]       Handle        SHELL_FILE_HANDLE to read from.
 | 
						|
 | 
						|
  @return                        The line of text from the file.
 | 
						|
  @retval NULL                   There was not enough memory available.
 | 
						|
 | 
						|
  @sa ShellFileHandleReadLine
 | 
						|
**/
 | 
						|
CHAR16*
 | 
						|
EFIAPI
 | 
						|
ParseReturnStdInLine (
 | 
						|
  IN SHELL_FILE_HANDLE Handle
 | 
						|
  )
 | 
						|
{
 | 
						|
  CHAR16          *RetVal;
 | 
						|
  UINTN           Size;
 | 
						|
  EFI_STATUS      Status;
 | 
						|
 | 
						|
  Size   = 0;
 | 
						|
  RetVal = NULL;
 | 
						|
 | 
						|
  Status = ShellFileHandleReadStdInLine (Handle, RetVal, &Size, FALSE);
 | 
						|
  if (Status == EFI_BUFFER_TOO_SMALL) {
 | 
						|
    RetVal = AllocateZeroPool(Size);
 | 
						|
    if (RetVal == NULL) {
 | 
						|
      return (NULL);
 | 
						|
    }
 | 
						|
    Status = ShellFileHandleReadStdInLine (Handle, RetVal, &Size, FALSE);
 | 
						|
 | 
						|
  }
 | 
						|
  if (EFI_ERROR(Status) && (RetVal != NULL)) {
 | 
						|
    FreePool(RetVal);
 | 
						|
    RetVal = NULL;
 | 
						|
  }
 | 
						|
  return (RetVal);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Handle stings for SFO Output with escape character ^ in a string
 | 
						|
  1. Quotation marks in the string must be escaped by using a ^ character (i.e. ^"). 
 | 
						|
  2. The ^ character may be inserted using ^^.
 | 
						|
 | 
						|
  @param[in]  String  The Unicode NULL-terminated string.
 | 
						|
 
 | 
						|
  @retval NewString   The new string handled for SFO.
 | 
						|
**/
 | 
						|
EFI_STRING
 | 
						|
HandleStringWithEscapeCharForParse (
 | 
						|
  IN      CHAR16  *String
 | 
						|
  )
 | 
						|
{
 | 
						|
  EFI_STRING  NewStr;
 | 
						|
  EFI_STRING  StrWalker;
 | 
						|
  EFI_STRING  ReturnStr;
 | 
						|
 | 
						|
  if (String == NULL) {
 | 
						|
    return NULL;
 | 
						|
  }
 | 
						|
  
 | 
						|
  //
 | 
						|
  // start to parse the input string.
 | 
						|
  //
 | 
						|
  NewStr = AllocateZeroPool (StrSize (String));
 | 
						|
  if (NewStr == NULL) {
 | 
						|
    return NULL;
 | 
						|
  }
 | 
						|
  ReturnStr = NewStr;
 | 
						|
  StrWalker = String;
 | 
						|
  while (*StrWalker != CHAR_NULL) {
 | 
						|
    if (*StrWalker == L'^' && (*(StrWalker + 1) == L'^' || *(StrWalker + 1) == L'"')) {
 | 
						|
      *NewStr = *(StrWalker + 1);
 | 
						|
      StrWalker++;
 | 
						|
    } else {
 | 
						|
      *NewStr = *StrWalker;
 | 
						|
    }
 | 
						|
    StrWalker++;
 | 
						|
    NewStr++;
 | 
						|
  }
 | 
						|
  
 | 
						|
  return ReturnStr;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Do the actual parsing of the file.  the file should be SFO output from a 
 | 
						|
  shell command or a similar format.
 | 
						|
 | 
						|
  @param[in] FileName               The filename to open.
 | 
						|
  @param[in] TableName              The name of the table to find.
 | 
						|
  @param[in] ColumnIndex            The column number to get.
 | 
						|
  @param[in] TableNameInstance      Which instance of the table to get (row).
 | 
						|
  @param[in] ShellCommandInstance   Which instance of the command to get.
 | 
						|
  @param[in] StreamingUnicode       Indicates Input file is StdIn Unicode streaming data or not
 | 
						|
 | 
						|
  @retval SHELL_NOT_FOUND     The requested instance was not found.
 | 
						|
  @retval SHELL_SUCCESS       The operation was successful.
 | 
						|
**/
 | 
						|
SHELL_STATUS
 | 
						|
EFIAPI
 | 
						|
PerformParsing(
 | 
						|
  IN CONST CHAR16 *FileName,
 | 
						|
  IN CONST CHAR16 *TableName,
 | 
						|
  IN CONST UINTN  ColumnIndex,
 | 
						|
  IN CONST UINTN  TableNameInstance,
 | 
						|
  IN CONST UINTN  ShellCommandInstance,
 | 
						|
  IN BOOLEAN      StreamingUnicode
 | 
						|
  )
 | 
						|
{
 | 
						|
  SHELL_FILE_HANDLE FileHandle;
 | 
						|
  EFI_STATUS        Status;
 | 
						|
  BOOLEAN           Ascii;
 | 
						|
  UINTN             LoopVariable;
 | 
						|
  UINTN             ColumnLoop;
 | 
						|
  CHAR16            *TempLine;
 | 
						|
  CHAR16            *ColumnPointer;
 | 
						|
  SHELL_STATUS      ShellStatus;
 | 
						|
  CHAR16            *TempSpot;
 | 
						|
  CHAR16            *SfoString;
 | 
						|
 | 
						|
  ASSERT(FileName   != NULL);
 | 
						|
  ASSERT(TableName  != NULL);
 | 
						|
 | 
						|
  ShellStatus       = SHELL_SUCCESS;
 | 
						|
 | 
						|
  Status = ShellOpenFileByName(FileName, &FileHandle, EFI_FILE_MODE_READ, 0);
 | 
						|
  if (EFI_ERROR(Status)) {
 | 
						|
    ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_FILE_OPEN_FAIL), gShellLevel2HiiHandle, L"parse", FileName);  
 | 
						|
    ShellStatus = SHELL_NOT_FOUND;
 | 
						|
  } else if (!EFI_ERROR (FileHandleIsDirectory (FileHandle))) {
 | 
						|
    ShellPrintHiiEx (-1, -1, NULL, STRING_TOKEN (STR_GEN_NOT_FILE), gShellLevel2HiiHandle, L"parse", FileName);  
 | 
						|
    ShellStatus = SHELL_NOT_FOUND;
 | 
						|
  } else {
 | 
						|
    for (LoopVariable = 0 ; LoopVariable < ShellCommandInstance && !ShellFileHandleEof(FileHandle);) {
 | 
						|
     if (StreamingUnicode) {
 | 
						|
       TempLine = ParseReturnStdInLine (FileHandle);
 | 
						|
     } else {
 | 
						|
       TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii); 
 | 
						|
     }
 | 
						|
 | 
						|
      if ((TempLine == NULL) || (*TempLine == CHAR_NULL && StreamingUnicode)) {
 | 
						|
         break;
 | 
						|
      }
 | 
						|
 | 
						|
      //
 | 
						|
      // Search for "ShellCommand," in the file to start the SFO table
 | 
						|
      // for a given ShellCommand.  The UEFI Shell spec does not specify
 | 
						|
      // a space after the comma.
 | 
						|
      //
 | 
						|
      if (StrStr (TempLine, L"ShellCommand,") == TempLine) {
 | 
						|
        LoopVariable++;
 | 
						|
      }
 | 
						|
      SHELL_FREE_NON_NULL(TempLine);
 | 
						|
    }
 | 
						|
    if (LoopVariable == ShellCommandInstance) {
 | 
						|
      LoopVariable = 0;
 | 
						|
      while(1) {
 | 
						|
        if (StreamingUnicode) {
 | 
						|
          TempLine = ParseReturnStdInLine (FileHandle);
 | 
						|
        } else {
 | 
						|
          TempLine = ShellFileHandleReturnLine (FileHandle, &Ascii); 
 | 
						|
        }
 | 
						|
        if (TempLine == NULL
 | 
						|
            || *TempLine == CHAR_NULL
 | 
						|
            || StrStr (TempLine, L"ShellCommand,") == TempLine) {
 | 
						|
          SHELL_FREE_NON_NULL(TempLine);
 | 
						|
          break;
 | 
						|
        }
 | 
						|
        if (StrStr (TempLine, TableName) == TempLine) {
 | 
						|
          LoopVariable++;
 | 
						|
          if (LoopVariable == TableNameInstance
 | 
						|
              || (TableNameInstance == (UINTN)-1)) {
 | 
						|
            for (ColumnLoop = 1, ColumnPointer = TempLine; ColumnLoop < ColumnIndex && ColumnPointer != NULL && *ColumnPointer != CHAR_NULL; ColumnLoop++) {
 | 
						|
              ColumnPointer = StrStr (ColumnPointer, L",\"");
 | 
						|
              if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL){
 | 
						|
                ColumnPointer++;
 | 
						|
              }
 | 
						|
            }
 | 
						|
            if (ColumnLoop == ColumnIndex) {
 | 
						|
              if (ColumnPointer == NULL) {
 | 
						|
                ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_NO_VALUE), gShellLevel2HiiHandle, L"parse", L"Column Index");  
 | 
						|
                ShellStatus = SHELL_INVALID_PARAMETER;
 | 
						|
              } else {
 | 
						|
                TempSpot = StrStr (ColumnPointer, L",\"");
 | 
						|
                if (TempSpot != NULL) {
 | 
						|
                  *TempSpot = CHAR_NULL;
 | 
						|
                }
 | 
						|
                while (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[0] == L' '){
 | 
						|
                  ColumnPointer++;
 | 
						|
                }
 | 
						|
                if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[0] == L'\"'){
 | 
						|
                  ColumnPointer++;
 | 
						|
                }
 | 
						|
                if (ColumnPointer != NULL && *ColumnPointer != CHAR_NULL && ColumnPointer[StrLen (ColumnPointer) - 1] == L'\"'){
 | 
						|
                  ColumnPointer[StrLen (ColumnPointer) - 1] = CHAR_NULL;
 | 
						|
                }
 | 
						|
                SfoString = HandleStringWithEscapeCharForParse (ColumnPointer);
 | 
						|
                if (SfoString != NULL) {
 | 
						|
                  ShellPrintEx (-1, -1, L"%s\r\n", SfoString);
 | 
						|
                  SHELL_FREE_NON_NULL (SfoString);
 | 
						|
                }
 | 
						|
              }
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
        SHELL_FREE_NON_NULL(TempLine);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return (ShellStatus);
 | 
						|
}
 | 
						|
 | 
						|
STATIC CONST SHELL_PARAM_ITEM ParamList[] = {
 | 
						|
  {L"-i", TypeValue},
 | 
						|
  {L"-s", TypeValue},
 | 
						|
  {NULL, TypeMax}
 | 
						|
  };
 | 
						|
 | 
						|
/**
 | 
						|
  Function for 'parse' command.
 | 
						|
 | 
						|
  @param[in] ImageHandle  Handle to the Image (NULL if Internal).
 | 
						|
  @param[in] SystemTable  Pointer to the System Table (NULL if Internal).
 | 
						|
**/
 | 
						|
SHELL_STATUS
 | 
						|
EFIAPI
 | 
						|
ShellCommandRunParse (
 | 
						|
  IN EFI_HANDLE        ImageHandle,
 | 
						|
  IN EFI_SYSTEM_TABLE  *SystemTable
 | 
						|
  )
 | 
						|
{
 | 
						|
  EFI_STATUS          Status;
 | 
						|
  LIST_ENTRY          *Package;
 | 
						|
  CHAR16              *ProblemParam;
 | 
						|
  CONST CHAR16        *FileName;
 | 
						|
  CONST CHAR16        *TableName;
 | 
						|
  CONST CHAR16        *ColumnString;
 | 
						|
  SHELL_STATUS        ShellStatus;
 | 
						|
  UINTN               ShellCommandInstance;
 | 
						|
  UINTN               TableNameInstance;
 | 
						|
  BOOLEAN             StreamingUnicode;
 | 
						|
 | 
						|
  ShellStatus      = SHELL_SUCCESS;
 | 
						|
  ProblemParam     = NULL;
 | 
						|
  StreamingUnicode = FALSE;
 | 
						|
 | 
						|
  //
 | 
						|
  // initialize the shell lib (we must be in non-auto-init...)
 | 
						|
  //
 | 
						|
  Status = ShellInitialize();
 | 
						|
  ASSERT_EFI_ERROR(Status);
 | 
						|
 | 
						|
  //
 | 
						|
  // parse the command line
 | 
						|
  //
 | 
						|
  Status = ShellCommandLineParseEx (ParamList, &Package, &ProblemParam, TRUE, FALSE);
 | 
						|
  if (EFI_ERROR(Status)) {
 | 
						|
    if (Status == EFI_VOLUME_CORRUPTED && ProblemParam != NULL) {
 | 
						|
      ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_PROBLEM), gShellLevel2HiiHandle, L"parse", ProblemParam);  
 | 
						|
      FreePool(ProblemParam);
 | 
						|
      ShellStatus = SHELL_INVALID_PARAMETER;
 | 
						|
    } else {
 | 
						|
      ASSERT(FALSE);
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    StreamingUnicode = IsStdInDataAvailable ();
 | 
						|
    if ((!StreamingUnicode && (ShellCommandLineGetCount(Package) < 4)) ||
 | 
						|
        (ShellCommandLineGetCount(Package) < 3)) {
 | 
						|
      ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_FEW), gShellLevel2HiiHandle, L"parse");  
 | 
						|
      ShellStatus = SHELL_INVALID_PARAMETER;
 | 
						|
    } else if ((StreamingUnicode && (ShellCommandLineGetCount(Package) > 3)) ||
 | 
						|
                (ShellCommandLineGetCount(Package) > 4)) {
 | 
						|
      ShellPrintHiiEx(-1, -1, NULL, STRING_TOKEN (STR_GEN_TOO_MANY), gShellLevel2HiiHandle, L"parse");  
 | 
						|
      ShellStatus = SHELL_INVALID_PARAMETER;
 | 
						|
    } else {
 | 
						|
      if (StreamingUnicode) {
 | 
						|
        FileName         = L">i";
 | 
						|
        TableName        = ShellCommandLineGetRawValue(Package, 1);
 | 
						|
        ColumnString     = ShellCommandLineGetRawValue(Package, 2);
 | 
						|
      } else {
 | 
						|
        FileName         = ShellCommandLineGetRawValue(Package, 1);
 | 
						|
        TableName        = ShellCommandLineGetRawValue(Package, 2);
 | 
						|
        ColumnString     = ShellCommandLineGetRawValue(Package, 3);
 | 
						|
      }
 | 
						|
      if (ShellCommandLineGetValue(Package, L"-i") == NULL) {
 | 
						|
        TableNameInstance = (UINTN)-1;
 | 
						|
      } else {
 | 
						|
        TableNameInstance = ShellStrToUintn(ShellCommandLineGetValue(Package, L"-i"));
 | 
						|
      }
 | 
						|
      if (ShellCommandLineGetValue(Package, L"-s") == NULL) {
 | 
						|
        ShellCommandInstance = 1;
 | 
						|
      } else {
 | 
						|
        ShellCommandInstance = ShellStrToUintn(ShellCommandLineGetValue(Package, L"-s"));
 | 
						|
      }
 | 
						|
 | 
						|
      ShellStatus = PerformParsing(FileName, TableName, ShellStrToUintn(ColumnString), TableNameInstance, ShellCommandInstance, StreamingUnicode);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // free the command line package
 | 
						|
  //
 | 
						|
  ShellCommandLineFreeVarList (Package);
 | 
						|
 | 
						|
  return (ShellStatus);
 | 
						|
}
 | 
						|
 |