blob: 7fe5a1102cab3fd3099d7226d847ca32709b264c [file] [log] [blame]
jcarseya405b862010-09-14 05:18:09 +00001/** @file
2 Main file for NULL named library for level 2 shell command functions.
3
4 these functions are:
5 attrib,
6 cd,
7 cp,
8 date*,
9 time*,
10 load,
11 ls,
12 map,
13 mkdir,
14 mv,
15 parse,
16 rm,
17 reset,
18 set,
19 timezone*
20
21 * functions are non-interactive only
22
23
24 Copyright (c) 2009 - 2010, Intel Corporation. All rights reserved.<BR>
25 This program and the accompanying materials
26 are licensed and made available under the terms and conditions of the BSD License
27 which accompanies this distribution. The full text of the license may be found at
28 http://opensource.org/licenses/bsd-license.php
29
30 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
31 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
32
33**/
34#include "UefiShellLevel2CommandsLib.h"
35
36CONST CHAR16 mFileName[] = L"ShellCommands";
37EFI_HANDLE gShellLevel2HiiHandle = NULL;
38CONST EFI_GUID gShellLevel2HiiGuid = \
39 { \
40 0xf95a7ccc, 0x4c55, 0x4426, { 0xa7, 0xb4, 0xdc, 0x89, 0x61, 0x95, 0xb, 0xae } \
41 };
42
43CONST CHAR16*
44EFIAPI
45ShellCommandGetManFileNameLevel2 (
46 VOID
47 )
48{
49 return (mFileName);
50}
51
52/**
53 Constructor for the Shell Level 2 Commands library.
54
55 Install the handlers for level 2 UEFI Shell 2.0 commands.
56
57 @param ImageHandle the image handle of the process
58 @param SystemTable the EFI System Table pointer
59
60 @retval EFI_SUCCESS the shell command handlers were installed sucessfully
61 @retval EFI_UNSUPPORTED the shell level required was not found.
62**/
63EFI_STATUS
64EFIAPI
65ShellLevel2CommandsLibConstructor (
66 IN EFI_HANDLE ImageHandle,
67 IN EFI_SYSTEM_TABLE *SystemTable
68 )
69{
70 //
71 // if shell level is less than 2 do nothing
72 //
73 if (PcdGet8(PcdShellSupportLevel) < 2) {
74 return (EFI_UNSUPPORTED);
75 }
76
77 gShellLevel2HiiHandle = HiiAddPackages (&gShellLevel2HiiGuid, gImageHandle, UefiShellLevel2CommandsLibStrings, NULL);
78 if (gShellLevel2HiiHandle == NULL) {
79 return (EFI_DEVICE_ERROR);
80 }
81
82 //
83 // install our shell command handlers that are always installed
84 //
85 ShellCommandRegisterCommandName(L"attrib", ShellCommandRunAttrib , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_ATTRIB) );
86 ShellCommandRegisterCommandName(L"cd", ShellCommandRunCd , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_CD) );
87 ShellCommandRegisterCommandName(L"cp", ShellCommandRunCp , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_CP) );
88 ShellCommandRegisterCommandName(L"load", ShellCommandRunLoad , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_LOAD) );
89 ShellCommandRegisterCommandName(L"map", ShellCommandRunMap , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_MAP) );
90 ShellCommandRegisterCommandName(L"mkdir", ShellCommandRunMkDir , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_MKDIR) );
91 ShellCommandRegisterCommandName(L"mv", ShellCommandRunMv , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_MV) );
92 ShellCommandRegisterCommandName(L"parse", ShellCommandRunParse , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_PARSE) );
93 ShellCommandRegisterCommandName(L"reset", ShellCommandRunReset , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_RESET) );
94 ShellCommandRegisterCommandName(L"set", ShellCommandRunSet , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_SET) );
95 ShellCommandRegisterCommandName(L"ls", ShellCommandRunLs , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_LS) );
96 ShellCommandRegisterCommandName(L"rm", ShellCommandRunRm , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_RM) );
97
98 //
99 // support for permenant (built in) aliases
100 //
101 ShellCommandRegisterAlias(L"rm", L"del");
102 ShellCommandRegisterAlias(L"ls", L"dir");
103 ShellCommandRegisterAlias(L"cp", L"copy");
104 ShellCommandRegisterAlias(L"mkdir", L"md");
105 ShellCommandRegisterAlias(L"cd ..", L"cd..");
106 ShellCommandRegisterAlias(L"cd \\", L"cd\\");
107 //
108 // These are installed in level 2 or 3...
109 //
110 if (PcdGet8(PcdShellSupportLevel) == 2 || PcdGet8(PcdShellSupportLevel) == 3) {
111 ShellCommandRegisterCommandName(L"date", ShellCommandRunDate , ShellCommandGetManFileNameLevel2, PcdGet8(PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_DATE) );
112 ShellCommandRegisterCommandName(L"time", ShellCommandRunTime , ShellCommandGetManFileNameLevel2, PcdGet8(PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_TIME) );
113 ShellCommandRegisterCommandName(L"timezone", ShellCommandRunTimeZone, ShellCommandGetManFileNameLevel2, PcdGet8(PcdShellSupportLevel), L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_TIMEZONE));
114 } else {
115 DEBUG_CODE_BEGIN();
116 //
117 // we want to be able to test these so install them under a different name in debug mode...
118 //
119 ShellCommandRegisterCommandName(L"l2date", ShellCommandRunDate , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_DATE) );
120 ShellCommandRegisterCommandName(L"l2time", ShellCommandRunTime , ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_TIME) );
121 ShellCommandRegisterCommandName(L"l2timezone", ShellCommandRunTimeZone, ShellCommandGetManFileNameLevel2, 2, L"", TRUE, gShellLevel2HiiHandle, STRING_TOKEN(STR_GET_HELP_TIMEZONE));
122 DEBUG_CODE_END();
123 }
124
125 return (EFI_SUCCESS);
126}
127
128/**
129 Destructor for the library. free any resources.
130
131 @param ImageHandle The image handle of the process.
132 @param SystemTable The EFI System Table pointer.
133
134 @retval EFI_SUCCESS Always returned.
135**/
136EFI_STATUS
137EFIAPI
138ShellLevel2CommandsLibDestructor (
139 IN EFI_HANDLE ImageHandle,
140 IN EFI_SYSTEM_TABLE *SystemTable
141 )
142{
143 if (gShellLevel2HiiHandle != NULL) {
144 HiiRemovePackages(gShellLevel2HiiHandle);
145 }
146 return (EFI_SUCCESS);
147}
148
149/**
150 Function to clean up paths. Removes the following items:
151 single periods in the path (no need for the current directory tag)
152 double periods in the path and removes a single parent directory.
153
154 This will be done inline and the resultant string may be be 'too big'.
155
156 @param[in] PathToReturn The pointer to the string containing the path.
157
158 @return PathToReturn is always returned.
159**/
160CHAR16*
161EFIAPI
162CleanPath(
163 IN CHAR16 *PathToReturn
164 )
165{
166 CHAR16 *TempString;
167 UINTN TempSize;
168 if (PathToReturn==NULL) {
169 return(NULL);
170 }
171 //
172 // Fix up the directory name
173 //
174 while ((TempString = StrStr(PathToReturn, L"\\..\\")) != NULL) {
175 *TempString = CHAR_NULL;
176 TempString += 4;
177 ChopLastSlash(PathToReturn);
178 TempSize = StrSize(TempString);
179 CopyMem(PathToReturn+StrLen(PathToReturn), TempString, TempSize);
180 }
181 if ((TempString = StrStr(PathToReturn, L"\\..")) != NULL && *(TempString + 3) == CHAR_NULL) {
182 *TempString = CHAR_NULL;
183 ChopLastSlash(PathToReturn);
184 }
185 while ((TempString = StrStr(PathToReturn, L"\\.\\")) != NULL) {
186 *TempString = CHAR_NULL;
187 TempString += 2;
188 TempSize = StrSize(TempString);
189 CopyMem(PathToReturn+StrLen(PathToReturn), TempString, TempSize);
190 }
191 if ((TempString = StrStr(PathToReturn, L"\\.")) != NULL && *(TempString + 2) == CHAR_NULL) {
192 *TempString = CHAR_NULL;
193 }
194 return (PathToReturn);
195}
196
197/**
198 returns a fully qualified directory (contains a map drive at the begining)
199 path from a unknown directory path.
200
201 If Path is already fully qualified this will return a duplicat otherwise this
202 will use get the current directory and use that to build the fully qualified
203 version.
204
205 if the return value is not NULL it must be caller freed.
206
207 @param[in] Path The unknown Path Value
208
209 @retval NULL A memory allocation failed
210 @retval NULL a fully qualified path could not be discovered.
211 @retval other pointer to a fuly qualified path.
212**/
213CHAR16*
214EFIAPI
215GetFullyQualifiedPath(
216 IN CONST CHAR16* Path
217 )
218{
219 CHAR16 *PathToReturn;
220 UINTN Size;
221 CONST CHAR16 *CurDir;
222
223 PathToReturn = NULL;
224 Size = 0;
225
226 ASSERT((PathToReturn == NULL && Size == 0) || (PathToReturn != NULL));
227 //
228 // convert a local path to an absolute path
229 //
230 if (StrStr(Path, L":") == NULL) {
231 CurDir = gEfiShellProtocol->GetCurDir(NULL);
232 StrnCatGrow(&PathToReturn, &Size, CurDir, 0);
233 if (*Path == L'\\') {
234 Path++;
235 }
236 }
237 StrnCatGrow(&PathToReturn, &Size, Path, 0);
238
239 CleanPath(PathToReturn);
240
241 while (PathToReturn[StrLen(PathToReturn)-1] == L'*') {
242 PathToReturn[StrLen(PathToReturn)-1] = CHAR_NULL;
243 }
244
245 return (PathToReturn);
246}
247
248/**
249 Function to verify all intermediate directories in the path.
250
251 @param[in] Path The pointer to the path to fix.
252
253 @retval EFI_SUCCESS The operation was successful.
254**/
255EFI_STATUS
256EFIAPI
257VerifyIntermediateDirectories (
258 IN CONST CHAR16 *Path
259 )
260{
261 EFI_STATUS Status;
262 CHAR16 *PathCopy;
263 CHAR16 *TempSpot;
264 SHELL_FILE_HANDLE FileHandle;
265
266 ASSERT(Path != NULL);
267
268 Status = EFI_SUCCESS;
269 PathCopy = NULL;
270 PathCopy = StrnCatGrow(&PathCopy, NULL, Path, 0);
271 FileHandle = NULL;
272
273 for (TempSpot = &PathCopy[StrLen(PathCopy)-1] ; *TempSpot != CHAR_NULL && *TempSpot != L'\\' ; TempSpot = &PathCopy[StrLen(PathCopy)-1]){
274 *TempSpot = CHAR_NULL;
275 }
276 if (*TempSpot == L'\\') {
277 *TempSpot = CHAR_NULL;
278 }
279
280 if (PathCopy != NULL && *PathCopy != CHAR_NULL) {
281 Status = VerifyIntermediateDirectories(PathCopy);
282
283 if (PathCopy[StrLen(PathCopy)-1] != L':') {
284 if (!EFI_ERROR(Status)) {
285 Status = ShellOpenFileByName(PathCopy, &FileHandle, EFI_FILE_MODE_READ, 0);
286 if (FileHandle != NULL) {
287 ShellCloseFile(&FileHandle);
288 }
289 }
290 }
291 }
292
293 SHELL_FREE_NON_NULL(PathCopy);
294
295 return (Status);
296}
297
298// be lazy and borrow from baselib.
299CHAR16
300EFIAPI
301InternalCharToUpper (
302 IN CONST CHAR16 Char
303 );
304
305CONST CHAR16*
306EFIAPI
307StrniCmp(
308 IN CONST CHAR16 *Source,
309 IN CONST CHAR16 *Target,
310 IN CONST UINTN Count
311 )
312{
313 UINTN LoopCount;
314 CHAR16 Char1;
315 CHAR16 Char2;
316
317 ASSERT(Source != NULL);
318 ASSERT(Target != NULL);
319
320 for (LoopCount = 0 ; LoopCount < Count ; LoopCount++) {
321 Char1 = InternalCharToUpper(Source[LoopCount]);
322 Char2 = InternalCharToUpper(Target[LoopCount]);
323 if (Char1 != Char2) {
324 return (&Source[LoopCount]);
325 }
326 }
327 return (NULL);
328}
329