• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2   Implementation for S3 Boot Script Saver state driver.
3 
4   Copyright (c) 2006 - 2016, Intel Corporation. All rights reserved.<BR>
5 
6   This program and the accompanying materials
7   are licensed and made available under the terms and conditions
8   of the BSD License which accompanies this distribution.  The
9   full text of the license may be found at
10   http://opensource.org/licenses/bsd-license.php
11 
12   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
13   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
14 
15 **/
16 #include "InternalS3SaveState.h"
17 
18 EFI_HANDLE                    mHandle = NULL;
19 EFI_S3_SAVE_STATE_PROTOCOL    mS3SaveState = {
20   BootScriptWrite,
21   BootScriptInsert,
22   BootScriptLabel,
23   BootScriptCompare
24   };
25 /**
26   Internal function to add IO write opcode to the table.
27 
28   @param  Marker                The variable argument list to get the opcode
29                                 and associated attributes.
30 
31   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
32   @retval EFI_SUCCESS           Opcode is added.
33 
34 **/
35 EFI_STATUS
BootScriptWriteIoWrite(IN VA_LIST Marker)36 BootScriptWriteIoWrite (
37   IN VA_LIST                       Marker
38   )
39 {
40   S3_BOOT_SCRIPT_LIB_WIDTH Width;
41   UINT64                Address;
42   UINTN                 Count;
43   UINT8                 *Buffer;
44 
45   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
46   Address     = VA_ARG (Marker, UINT64);
47   Count       = VA_ARG (Marker, UINTN);
48   Buffer      = VA_ARG (Marker, UINT8 *);
49 
50   return S3BootScriptSaveIoWrite (Width, Address, Count, Buffer);
51 }
52 /**
53   Internal function to add IO read/write opcode to the table.
54 
55   @param  Marker                The variable argument list to get the opcode
56                                 and associated attributes.
57 
58   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
59   @retval EFI_SUCCESS           Opcode is added.
60 
61 **/
62 EFI_STATUS
BootScriptWriteIoReadWrite(IN VA_LIST Marker)63 BootScriptWriteIoReadWrite (
64   IN VA_LIST                       Marker
65   )
66 {
67   S3_BOOT_SCRIPT_LIB_WIDTH Width;
68   UINT64                Address;
69   UINT8                 *Data;
70   UINT8                 *DataMask;
71 
72   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
73   Address     = VA_ARG (Marker, UINT64);
74   Data        = VA_ARG (Marker, UINT8 *);
75   DataMask    = VA_ARG (Marker, UINT8 *);
76 
77   return S3BootScriptSaveIoReadWrite (Width, Address, Data, DataMask);
78 }
79 
80 /**
81   Internal function to add memory write opcode to the table.
82 
83   @param  Marker                The variable argument list to get the opcode
84                                 and associated attributes.
85 
86   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
87   @retval EFI_SUCCESS           Opcode is added.
88 
89 **/
90 EFI_STATUS
BootScriptWriteMemWrite(IN VA_LIST Marker)91 BootScriptWriteMemWrite (
92   IN VA_LIST                       Marker
93   )
94 {
95   S3_BOOT_SCRIPT_LIB_WIDTH Width;
96   UINT64                Address;
97   UINTN                 Count;
98   UINT8                 *Buffer;
99 
100   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
101   Address     = VA_ARG (Marker, UINT64);
102   Count       = VA_ARG (Marker, UINTN);
103   Buffer      = VA_ARG (Marker, UINT8 *);
104 
105   return S3BootScriptSaveMemWrite (Width, Address, Count, Buffer);
106 }
107 
108 /**
109   Internal function to add memory read/write opcode to the table.
110 
111   @param  Marker                The variable argument list to get the opcode
112                                 and associated attributes.
113 
114   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
115   @retval EFI_SUCCESS           Opcode is added.
116 
117 **/
118 EFI_STATUS
BootScriptWriteMemReadWrite(IN VA_LIST Marker)119 BootScriptWriteMemReadWrite (
120   IN VA_LIST                       Marker
121   )
122 {
123   S3_BOOT_SCRIPT_LIB_WIDTH Width;
124   UINT64                Address;
125   UINT8                 *Data;
126   UINT8                 *DataMask;
127 
128   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
129   Address     = VA_ARG (Marker, UINT64);
130   Data        = VA_ARG (Marker, UINT8 *);
131   DataMask    = VA_ARG (Marker, UINT8 *);
132 
133   return S3BootScriptSaveMemReadWrite (Width, Address, Data, DataMask);
134 }
135 
136 /**
137   Internal function to add PciCfg write opcode to the table.
138 
139   @param  Marker                The variable argument list to get the opcode
140                                 and associated attributes.
141 
142   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
143   @retval EFI_SUCCESS           Opcode is added.
144 
145 **/
146 EFI_STATUS
BootScriptWritePciCfgWrite(IN VA_LIST Marker)147 BootScriptWritePciCfgWrite (
148   IN VA_LIST                       Marker
149   )
150 {
151   S3_BOOT_SCRIPT_LIB_WIDTH Width;
152   UINT64                Address;
153   UINTN                 Count;
154   UINT8                 *Buffer;
155 
156   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
157   Address     = VA_ARG (Marker, UINT64);
158   Count       = VA_ARG (Marker, UINTN);
159   Buffer      = VA_ARG (Marker, UINT8 *);
160 
161   return S3BootScriptSavePciCfgWrite (Width, Address, Count, Buffer);
162 }
163 
164 /**
165   Internal function to PciCfg read/write opcode to the table.
166 
167   @param  Marker                The variable argument list to get the opcode
168                                 and associated attributes.
169 
170   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
171   @retval EFI_SUCCESS           Opcode is added.
172 
173 **/
174 EFI_STATUS
BootScriptWritePciCfgReadWrite(IN VA_LIST Marker)175 BootScriptWritePciCfgReadWrite (
176   IN VA_LIST                       Marker
177   )
178 {
179   S3_BOOT_SCRIPT_LIB_WIDTH Width;
180   UINT64                Address;
181   UINT8                 *Data;
182   UINT8                 *DataMask;
183 
184   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
185   Address     = VA_ARG (Marker, UINT64);
186   Data        = VA_ARG (Marker, UINT8 *);
187   DataMask    = VA_ARG (Marker, UINT8 *);
188 
189   return S3BootScriptSavePciCfgReadWrite (Width, Address, Data, DataMask);
190 }
191 /**
192   Internal function to add PciCfg2 write opcode to the table.
193 
194   @param  Marker                The variable argument list to get the opcode
195                                 and associated attributes.
196 
197   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
198   @retval EFI_SUCCESS           Opcode is added.
199 
200 **/
201 EFI_STATUS
BootScriptWritePciCfg2Write(IN VA_LIST Marker)202 BootScriptWritePciCfg2Write (
203   IN VA_LIST                       Marker
204   )
205 {
206   S3_BOOT_SCRIPT_LIB_WIDTH Width;
207   UINT64                Address;
208   UINTN                 Count;
209   UINT8                 *Buffer;
210   UINT16                Segment;
211 
212   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
213   Address     = VA_ARG (Marker, UINT64);
214   Count       = VA_ARG (Marker, UINTN);
215   Buffer      = VA_ARG (Marker, UINT8 *);
216   Segment     = VA_ARG (Marker, UINT16);
217 
218   return S3BootScriptSavePciCfg2Write (Width, Segment, Address, Count, Buffer);
219 }
220 
221 /**
222   Internal function to PciCfg2 read/write opcode to the table.
223 
224   @param  Marker                The variable argument list to get the opcode
225                                 and associated attributes.
226 
227   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
228   @retval EFI_SUCCESS           Opcode is added.
229 
230 **/
231 EFI_STATUS
BootScriptWritePciCfg2ReadWrite(IN VA_LIST Marker)232 BootScriptWritePciCfg2ReadWrite (
233   IN VA_LIST                       Marker
234   )
235 {
236   S3_BOOT_SCRIPT_LIB_WIDTH Width;
237   UINT16                Segment;
238   UINT64                Address;
239   UINT8                 *Data;
240   UINT8                 *DataMask;
241 
242   Width       = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
243   Address     = VA_ARG (Marker, UINT64);
244   Segment     = VA_ARG (Marker, UINT16);
245   Data        = VA_ARG (Marker, UINT8 *);
246   DataMask    = VA_ARG (Marker, UINT8 *);
247 
248   return S3BootScriptSavePciCfg2ReadWrite (Width, Segment, Address, Data, DataMask);
249 }
250 /**
251   Internal function to add smbus execute opcode to the table.
252 
253   @param  Marker                The variable argument list to get the opcode
254                                 and associated attributes.
255 
256   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
257   @retval EFI_SUCCESS           Opcode is added.
258 
259 **/
260 EFI_STATUS
BootScriptWriteSmbusExecute(IN VA_LIST Marker)261 BootScriptWriteSmbusExecute (
262   IN VA_LIST                       Marker
263   )
264 {
265   EFI_SMBUS_DEVICE_ADDRESS  SlaveAddress;
266   EFI_SMBUS_DEVICE_COMMAND  Command;
267   EFI_SMBUS_OPERATION       Operation;
268   BOOLEAN                   PecCheck;
269   VOID                     *Buffer;
270   UINTN                    *DataSize;
271   UINTN                     SmBusAddress;
272 
273   SlaveAddress.SmbusDeviceAddress = VA_ARG (Marker, UINTN);
274   Command                         = VA_ARG (Marker, EFI_SMBUS_DEVICE_COMMAND);
275   Operation                       = VA_ARG (Marker, EFI_SMBUS_OPERATION);
276   PecCheck                        = VA_ARG (Marker, BOOLEAN);
277   SmBusAddress                    = SMBUS_LIB_ADDRESS (SlaveAddress.SmbusDeviceAddress,Command,0,PecCheck);
278   DataSize                        = VA_ARG (Marker, UINTN *);
279   Buffer                          = VA_ARG (Marker, VOID *);
280 
281   return S3BootScriptSaveSmbusExecute (SmBusAddress, Operation, DataSize, Buffer);
282 }
283 /**
284   Internal function to add stall opcode to the table.
285 
286   @param  Marker                The variable argument list to get the opcode
287                                 and associated attributes.
288 
289   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
290   @retval EFI_SUCCESS           Opcode is added.
291 
292 **/
293 EFI_STATUS
BootScriptWriteStall(IN VA_LIST Marker)294 BootScriptWriteStall (
295   IN VA_LIST                       Marker
296   )
297 {
298   UINT32                Duration;
299 
300   Duration    = VA_ARG (Marker, UINT32);
301 
302   return S3BootScriptSaveStall (Duration);
303 }
304 
305 /**
306   Internal function to add Save jmp address according to DISPATCH_OPCODE.
307   We ignore "Context" parameter
308 
309   @param  Marker                The variable argument list to get the opcode
310                                 and associated attributes.
311 
312   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
313   @retval EFI_SUCCESS           Opcode is added.
314 
315 **/
316 EFI_STATUS
BootScriptWriteDispatch(IN VA_LIST Marker)317 BootScriptWriteDispatch (
318   IN VA_LIST                       Marker
319   )
320 {
321   VOID        *EntryPoint;
322 
323   EntryPoint = (VOID*)(UINTN)VA_ARG (Marker, EFI_PHYSICAL_ADDRESS);
324   return S3BootScriptSaveDispatch (EntryPoint);
325 }
326 
327 /**
328   Internal function to add memory pool operation to the table.
329 
330   @param  Marker                The variable argument list to get the opcode
331                                 and associated attributes.
332 
333   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
334   @retval EFI_SUCCESS           Opcode is added.
335 
336 **/
337 EFI_STATUS
BootScriptWriteMemPoll(IN VA_LIST Marker)338 BootScriptWriteMemPoll (
339   IN VA_LIST                       Marker
340   )
341 {
342   S3_BOOT_SCRIPT_LIB_WIDTH   Width;
343   UINT64                     Address;
344   VOID                      *Data;
345   VOID                      *DataMask;
346   UINT64                    Delay;
347   UINT64                    LoopTimes;
348   UINT32                    Remainder;
349 
350   Width    = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
351   Address  = VA_ARG (Marker, UINT64);
352   Data     = VA_ARG (Marker, VOID *);
353   DataMask = VA_ARG (Marker, VOID *);
354   Delay    = VA_ARG (Marker, UINT64);
355   //
356   // According to the spec, the interval between 2 polls is 100ns,
357   // but the unit of Duration for S3BootScriptSaveMemPoll() is microsecond(1000ns).
358   // Duration * 1000ns * LoopTimes = Delay * 100ns
359   // Duration will be minimum 1(microsecond) to be minimum deviation,
360   // so LoopTimes = Delay / 10.
361   //
362   LoopTimes = DivU64x32Remainder (
363                 Delay,
364                 10,
365                 &Remainder
366                 );
367   if (Remainder != 0) {
368     //
369     // If Remainder is not zero, LoopTimes will be rounded up by 1.
370     //
371     LoopTimes +=1;
372   }
373   return S3BootScriptSaveMemPoll (Width, Address, DataMask, Data, 1, LoopTimes);
374 
375 }
376 
377 /**
378   Internal function to add Save jmp address according to DISPATCH_OPCODE2.
379   The "Context" parameter is not ignored.
380 
381   @param  Marker                The variable argument list to get the opcode
382                                 and associated attributes.
383 
384   @retval EFI_OUT_OF_RESOURCES  Not enough resource to do operation.
385   @retval EFI_SUCCESS           Opcode is added.
386 
387 **/
388 EFI_STATUS
BootScriptWriteDispatch2(IN VA_LIST Marker)389 BootScriptWriteDispatch2 (
390   IN VA_LIST                       Marker
391   )
392 {
393   VOID                  *EntryPoint;
394   VOID                  *Context;
395 
396   EntryPoint = (VOID*)(UINTN)VA_ARG (Marker, EFI_PHYSICAL_ADDRESS);
397   Context    = (VOID*)(UINTN)VA_ARG (Marker, EFI_PHYSICAL_ADDRESS);
398 
399   return S3BootScriptSaveDispatch2 (EntryPoint, Context);
400 }
401 /**
402   Internal function to add INFORAMTION opcode node to the table
403   list.
404   @param  Marker                The variable argument list to get the opcode
405                                 and associated attributes.
406 
407   @retval EFI_OUT_OF_RESOURCES  Not enought resource to complete the operations.
408   @retval EFI_SUCCESS           The opcode entry is added to the  table
409                                 successfully.
410 **/
411 EFI_STATUS
BootScriptWriteInformation(IN VA_LIST Marker)412 BootScriptWriteInformation (
413   IN VA_LIST                       Marker
414   )
415 {
416   UINT32                InformationLength;
417   EFI_PHYSICAL_ADDRESS  Information;
418 
419   InformationLength = VA_ARG (Marker, UINT32);
420   Information = VA_ARG (Marker, EFI_PHYSICAL_ADDRESS);
421   return S3BootScriptSaveInformation (InformationLength, (VOID*)(UINTN)Information);
422 }
423 /**
424   Internal function to add IO poll opcode node  to the table
425   @param  Marker                The variable argument list to get the opcode
426                                 and associated attributes.
427 
428   @retval EFI_OUT_OF_RESOURCES  Not enought resource to complete the operations.
429   @retval EFI_SUCCESS           The opcode entry is added to the  table
430                                 successfully.
431 **/
432 EFI_STATUS
BootScriptWriteIoPoll(IN VA_LIST Marker)433 BootScriptWriteIoPoll (
434   IN VA_LIST                       Marker
435   )
436 {
437    S3_BOOT_SCRIPT_LIB_WIDTH     Width;
438    UINT64                     Address;
439    VOID                      *Data;
440    VOID                      *DataMask;
441    UINT64                     Delay;
442 
443    Width    = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
444    Address  = VA_ARG (Marker, UINT64);
445    Data     = VA_ARG (Marker, VOID *);
446    DataMask = VA_ARG (Marker, VOID *);
447    Delay    = (UINT64)VA_ARG (Marker, UINT64);
448 
449    return S3BootScriptSaveIoPoll (Width, Address, Data, DataMask, Delay);
450 }
451 /**
452   Internal function to add PCI config poll opcode node to the table
453 
454   @param  Marker                The variable argument list to get the opcode
455                                 and associated attributes.
456 
457   @retval EFI_OUT_OF_RESOURCES  Not enought resource to complete the operations.
458   @retval EFI_SUCCESS           The opcode entry is added to the  table
459                                 successfully.
460 **/
461 EFI_STATUS
BootScriptWritePciConfigPoll(IN VA_LIST Marker)462 BootScriptWritePciConfigPoll (
463   IN VA_LIST                       Marker
464   )
465 {
466    S3_BOOT_SCRIPT_LIB_WIDTH   Width;
467    UINT64                     Address;
468    VOID                      *Data;
469    VOID                      *DataMask;
470    UINT64                     Delay;
471 
472 
473    Width    = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
474    Address  = VA_ARG (Marker, UINT64);
475    Data     = VA_ARG (Marker, VOID *);
476    DataMask = VA_ARG (Marker, VOID *);
477    Delay    = (UINT64)VA_ARG (Marker, UINT64);
478 
479    return S3BootScriptSavePciPoll (Width, Address, Data, DataMask, Delay);
480 }
481 /**
482   Internal function to add PCI config 2 poll opcode node to the table
483 
484   @param  Marker                The variable argument list to get the opcode
485                                 and associated attributes.
486 
487   @retval EFI_OUT_OF_RESOURCES  Not enought resource to complete the operations.
488   @retval EFI_SUCCESS           The opcode entry is added to the  table
489                                 successfully.
490 **/
491 EFI_STATUS
BootScriptWritePciConfig2Poll(IN VA_LIST Marker)492 BootScriptWritePciConfig2Poll (
493   IN VA_LIST                       Marker
494   )
495 {
496    S3_BOOT_SCRIPT_LIB_WIDTH      Width;
497    UINT16                        Segment;
498    UINT64                        Address;
499    VOID                         *Data;
500    VOID                         *DataMask;
501    UINT64                        Delay;
502 
503    Width    = VA_ARG (Marker, S3_BOOT_SCRIPT_LIB_WIDTH);
504    Segment  = VA_ARG (Marker, UINT16);
505    Address  = VA_ARG (Marker, UINT64);
506    Data     = VA_ARG (Marker, VOID *);
507    DataMask = VA_ARG (Marker, VOID *);
508    Delay    = (UINT64)VA_ARG (Marker, UINT64);
509 
510    return S3BootScriptSavePci2Poll (Width, Segment, Address, Data, DataMask, Delay);
511 }
512 
513 
514 /**
515   Adds a record into S3 boot script table.
516 
517   This function is used to store a boot script record into a given boot
518   script table. If the table specified by TableName is nonexistent in the
519   system, a new table will automatically be created and then the script record
520   will be added into the new table. This function is responsible for allocating
521   necessary memory for the script.
522 
523   This function has a variable parameter list. The exact parameter list depends on
524   the OpCode that is passed into the function. If an unsupported OpCode or illegal
525   parameter list is passed in, this function returns EFI_INVALID_PARAMETER.
526   If there are not enough resources available for storing more scripts, this function returns
527   EFI_OUT_OF_RESOURCES.
528 
529   @param  This                  A pointer to the EFI_S3_SAVE_STATE_PROTOCOL instance.
530   @param  OpCode                The operation code (opcode) number.
531   @param  ...                   Argument list that is specific to each opcode.
532 
533   @retval EFI_SUCCESS           The operation succeeded. A record was added into the
534                                 specified script table.
535   @retval EFI_INVALID_PARAMETER The parameter is illegal or the given boot script is not supported.
536                                 If the opcode is unknow or not supported because of the PCD
537                                 Feature Flags.
538   @retval EFI_OUT_OF_RESOURCES  There is insufficient memory to store the boot script.
539 
540 **/
541 EFI_STATUS
542 EFIAPI
BootScriptWrite(IN CONST EFI_S3_SAVE_STATE_PROTOCOL * This,IN UINT16 OpCode,...)543 BootScriptWrite (
544   IN CONST EFI_S3_SAVE_STATE_PROTOCOL      *This,
545   IN       UINT16                           OpCode,
546   ...
547   )
548 {
549   EFI_STATUS                Status;
550   VA_LIST                   Marker;
551   //
552   // Build script according to opcode
553   //
554   switch (OpCode) {
555 
556   case EFI_BOOT_SCRIPT_IO_WRITE_OPCODE:
557     VA_START (Marker, OpCode);
558     Status = BootScriptWriteIoWrite (Marker);
559     VA_END (Marker);
560     break;
561 
562   case EFI_BOOT_SCRIPT_IO_READ_WRITE_OPCODE:
563     VA_START (Marker, OpCode);
564     Status = BootScriptWriteIoReadWrite (Marker);
565     VA_END (Marker);
566     break;
567 
568   case EFI_BOOT_SCRIPT_MEM_WRITE_OPCODE:
569     VA_START (Marker, OpCode);
570     Status = BootScriptWriteMemWrite (Marker);
571     VA_END (Marker);
572     break;
573 
574   case EFI_BOOT_SCRIPT_MEM_READ_WRITE_OPCODE:
575     VA_START (Marker, OpCode);
576     Status = BootScriptWriteMemReadWrite (Marker);
577     VA_END (Marker);
578     break;
579 
580   case EFI_BOOT_SCRIPT_PCI_CONFIG_WRITE_OPCODE:
581     VA_START (Marker, OpCode);
582     Status = BootScriptWritePciCfgWrite (Marker);
583     VA_END (Marker);
584     break;
585 
586   case EFI_BOOT_SCRIPT_PCI_CONFIG_READ_WRITE_OPCODE:
587     VA_START (Marker, OpCode);
588     Status = BootScriptWritePciCfgReadWrite (Marker);
589     VA_END (Marker);
590     break;
591 
592   case EFI_BOOT_SCRIPT_SMBUS_EXECUTE_OPCODE:
593     VA_START (Marker, OpCode);
594     Status = BootScriptWriteSmbusExecute (Marker);
595     VA_END (Marker);
596     break;
597 
598   case EFI_BOOT_SCRIPT_STALL_OPCODE:
599     VA_START (Marker, OpCode);
600     Status = BootScriptWriteStall (Marker);
601     VA_END (Marker);
602 
603     break;
604 
605   case EFI_BOOT_SCRIPT_DISPATCH_OPCODE:
606     VA_START (Marker, OpCode);
607     Status = BootScriptWriteDispatch (Marker);
608     VA_END (Marker);
609     break;
610 
611   case EFI_BOOT_SCRIPT_DISPATCH_2_OPCODE:
612     VA_START (Marker, OpCode);
613     Status = BootScriptWriteDispatch2 (Marker);
614     VA_END (Marker);
615     break;
616 
617   case EFI_BOOT_SCRIPT_INFORMATION_OPCODE:
618     VA_START (Marker, OpCode);
619     Status = BootScriptWriteInformation (Marker);
620     VA_END (Marker);
621     break;
622 
623   case EFI_BOOT_SCRIPT_MEM_POLL_OPCODE:
624     VA_START (Marker, OpCode);
625     Status = BootScriptWriteMemPoll (Marker);
626     VA_END (Marker);
627     break;
628 
629   case EFI_BOOT_SCRIPT_PCI_CONFIG2_WRITE_OPCODE:
630     VA_START (Marker, OpCode);
631     Status = BootScriptWritePciCfg2Write (Marker);
632     VA_END (Marker);
633     break;
634 
635   case EFI_BOOT_SCRIPT_PCI_CONFIG2_READ_WRITE_OPCODE:
636     VA_START (Marker, OpCode);
637     Status = BootScriptWritePciCfg2ReadWrite (Marker);
638     VA_END (Marker);
639     break;
640 
641   case EFI_BOOT_SCRIPT_IO_POLL_OPCODE:
642     VA_START (Marker, OpCode);
643     Status = BootScriptWriteIoPoll (Marker);
644     VA_END (Marker);
645     break;
646 
647   case EFI_BOOT_SCRIPT_PCI_CONFIG_POLL_OPCODE:
648     VA_START (Marker, OpCode);
649     Status = BootScriptWritePciConfigPoll (Marker);
650     VA_END (Marker);
651     break;
652 
653   case EFI_BOOT_SCRIPT_PCI_CONFIG2_POLL_OPCODE:
654     VA_START (Marker, OpCode);
655     Status = BootScriptWritePciConfig2Poll (Marker);
656     VA_END (Marker);
657     break;
658 
659   default:
660     Status = EFI_INVALID_PARAMETER;
661     break;
662   }
663 
664   return Status;
665 }
666 /**
667   Insert a record into a specified Framework boot script table.
668 
669   This function is used to store an OpCode to be replayed as part of the S3 resume boot path. It is
670   assumed this protocol has platform specific mechanism to store the OpCode set and replay them
671   during the S3 resume.
672   The opcode is inserted before or after the specified position in the boot script table. If Position is
673   NULL then that position is after the last opcode in the table (BeforeOrAfter is FALSE) or before
674   the first opcode in the table (BeforeOrAfter is TRUE). The position which is pointed to by
675   Position upon return can be used for subsequent insertions.
676 
677   @param  This                  A pointer to the EFI_S3_SAVE_STATE_PROTOCOL instance.
678   @param  BeforeOrAfter         Specifies whether the opcode is stored before (TRUE) or after (FALSE) the position
679                                 in the boot script table specified by Position. If Position is NULL or points to
680                                 NULL then the new opcode is inserted at the beginning of the table (if TRUE) or end
681                                 of the table (if FALSE).
682   @param  Position              On entry, specifies the position in the boot script table where the opcode will be
683                                 inserted, either before or after, depending on BeforeOrAfter. On exit, specifies
684                                 the position of the inserted opcode in the boot script table.
685   @param  OpCode                The operation code (opcode) number.
686   @param  ...                   Argument list that is specific to each opcode.
687 
688   @retval EFI_SUCCESS           The operation succeeded. A record was added into the
689                                 specified script table.
690   @retval EFI_INVALID_PARAMETER The Opcode is an invalid opcode value or the Position is not a valid position in the boot script table..
691   @retval EFI_OUT_OF_RESOURCES  There is insufficient memory to store the boot script.
692 
693 **/
694 EFI_STATUS
695 EFIAPI
BootScriptInsert(IN CONST EFI_S3_SAVE_STATE_PROTOCOL * This,IN BOOLEAN BeforeOrAfter,IN OUT EFI_S3_BOOT_SCRIPT_POSITION * Position OPTIONAL,IN UINT16 OpCode,...)696 BootScriptInsert (
697   IN CONST EFI_S3_SAVE_STATE_PROTOCOL     *This,
698   IN       BOOLEAN                        BeforeOrAfter,
699   IN OUT   EFI_S3_BOOT_SCRIPT_POSITION    *Position OPTIONAL,
700   IN       UINT16                         OpCode,
701   ...
702   )
703 {
704   EFI_STATUS                Status;
705   VA_LIST                   Marker;
706   //
707   // Build script according to opcode
708   //
709   switch (OpCode) {
710 
711   case EFI_BOOT_SCRIPT_IO_WRITE_OPCODE:
712     VA_START (Marker, OpCode);
713     Status = BootScriptWriteIoWrite (Marker);
714     VA_END (Marker);
715     break;
716 
717   case EFI_BOOT_SCRIPT_IO_READ_WRITE_OPCODE:
718     VA_START (Marker, OpCode);
719     Status = BootScriptWriteIoReadWrite (Marker);
720     VA_END (Marker);
721     break;
722 
723   case EFI_BOOT_SCRIPT_MEM_WRITE_OPCODE:
724     VA_START (Marker, OpCode);
725     Status = BootScriptWriteMemWrite (Marker);
726     VA_END (Marker);
727     break;
728 
729   case EFI_BOOT_SCRIPT_MEM_READ_WRITE_OPCODE:
730     VA_START (Marker, OpCode);
731     Status = BootScriptWriteMemReadWrite (Marker);
732     VA_END (Marker);
733     break;
734 
735   case EFI_BOOT_SCRIPT_PCI_CONFIG_WRITE_OPCODE:
736     VA_START (Marker, OpCode);
737     Status = BootScriptWritePciCfgWrite (Marker);
738     VA_END (Marker);
739     break;
740 
741   case EFI_BOOT_SCRIPT_PCI_CONFIG_READ_WRITE_OPCODE:
742     VA_START (Marker, OpCode);
743     Status = BootScriptWritePciCfgReadWrite (Marker);
744     VA_END (Marker);
745     break;
746 
747   case EFI_BOOT_SCRIPT_SMBUS_EXECUTE_OPCODE:
748     VA_START (Marker, OpCode);
749     Status = BootScriptWriteSmbusExecute (Marker);
750     VA_END (Marker);
751     break;
752 
753   case EFI_BOOT_SCRIPT_STALL_OPCODE:
754     VA_START (Marker, OpCode);
755     Status = BootScriptWriteStall (Marker);
756     VA_END (Marker);
757 
758     break;
759 
760   case EFI_BOOT_SCRIPT_DISPATCH_OPCODE:
761     VA_START (Marker, OpCode);
762     Status = BootScriptWriteDispatch (Marker);
763     VA_END (Marker);
764     break;
765 
766   case EFI_BOOT_SCRIPT_DISPATCH_2_OPCODE:
767     VA_START (Marker, OpCode);
768     Status = BootScriptWriteDispatch2 (Marker);
769     VA_END (Marker);
770     break;
771 
772   case EFI_BOOT_SCRIPT_INFORMATION_OPCODE:
773     VA_START (Marker, OpCode);
774     Status = BootScriptWriteInformation (Marker);
775     VA_END (Marker);
776     break;
777 
778   case EFI_BOOT_SCRIPT_MEM_POLL_OPCODE:
779     VA_START (Marker, OpCode);
780     Status = BootScriptWriteMemPoll (Marker);
781     VA_END (Marker);
782     break;
783 
784   case EFI_BOOT_SCRIPT_PCI_CONFIG2_WRITE_OPCODE:
785     VA_START (Marker, OpCode);
786     Status = BootScriptWritePciCfg2Write (Marker);
787     VA_END (Marker);
788     break;
789 
790   case EFI_BOOT_SCRIPT_PCI_CONFIG2_READ_WRITE_OPCODE:
791     VA_START (Marker, OpCode);
792     Status = BootScriptWritePciCfg2ReadWrite (Marker);
793     VA_END (Marker);
794     break;
795 
796   case EFI_BOOT_SCRIPT_IO_POLL_OPCODE:
797     VA_START (Marker, OpCode);
798     Status = BootScriptWriteIoPoll (Marker);
799     VA_END (Marker);
800     break;
801 
802   case EFI_BOOT_SCRIPT_PCI_CONFIG_POLL_OPCODE:
803     VA_START (Marker, OpCode);
804     Status = BootScriptWritePciConfigPoll (Marker);
805     VA_END (Marker);
806     break;
807 
808   case EFI_BOOT_SCRIPT_PCI_CONFIG2_POLL_OPCODE:
809     VA_START (Marker, OpCode);
810     Status = BootScriptWritePciConfig2Poll (Marker);
811     VA_END (Marker);
812     break;
813 
814   default:
815     Status = EFI_INVALID_PARAMETER;
816     break;
817   }
818 
819   if (!EFI_ERROR (Status)) {
820    Status = S3BootScriptMoveLastOpcode (BeforeOrAfter, Position);
821   }
822   return Status;
823 }
824 /**
825   Find a label within the boot script table and, if not present, optionally create it.
826 
827   If the label Label is already exists in the boot script table, then no new label is created, the
828   position of the Label is returned in *Position and EFI_SUCCESS is returned.
829   If the label Label does not already exist and CreateIfNotFound is TRUE, then it will be
830   created before or after the specified position and EFI_SUCCESS is returned.
831   If the label Label does not already exist and CreateIfNotFound is FALSE, then
832   EFI_NOT_FOUND is returned.
833 
834   @param  This                  A pointer to the EFI_S3_SAVE_STATE_PROTOCOL instance.
835   @param  BeforeOrAfter         Specifies whether the label is stored before (TRUE) or after (FALSE) the position in
836                                 the boot script table specified by Position. If Position is NULL or points to
837                                 NULL then the new label is inserted at the beginning of the table (if TRUE) or end of
838                                 the table (if FALSE).
839   @param  CreateIfNotFound      Specifies whether the label will be created if the label does not exists (TRUE) or not
840                                 (FALSE).
841   @param  Position              On entry, specifies the position in the boot script table where the label will be inserted,
842                                 either before or after, depending on BeforeOrAfter. On exit, specifies the position
843                                 of the inserted label in the boot script table.
844   @param  Label                 Points to the label which will be inserted in the boot script table.
845 
846   @retval EFI_SUCCESS           The label already exists or was inserted.
847   @retval EFI_INVALID_PARAMETER The Label is NULL or points to an empty string.
848   @retval EFI_INVALID_PARAMETER The Position is not a valid position in the boot script table.
849 
850 **/
851 EFI_STATUS
852 EFIAPI
BootScriptLabel(IN CONST EFI_S3_SAVE_STATE_PROTOCOL * This,IN BOOLEAN BeforeOrAfter,IN BOOLEAN CreateIfNotFound,IN OUT EFI_S3_BOOT_SCRIPT_POSITION * Position OPTIONAL,IN CONST CHAR8 * Label)853 BootScriptLabel (
854   IN CONST EFI_S3_SAVE_STATE_PROTOCOL           *This,
855   IN       BOOLEAN                              BeforeOrAfter,
856   IN       BOOLEAN                              CreateIfNotFound,
857   IN OUT   EFI_S3_BOOT_SCRIPT_POSITION          *Position OPTIONAL,
858   IN CONST CHAR8                                *Label
859   )
860 {
861 	return S3BootScriptLabel (BeforeOrAfter, CreateIfNotFound, Position, Label);
862 }
863 /**
864   Compare two positions in the boot script table and return their relative position.
865 
866   This function compares two positions in the boot script table and returns their relative positions. If
867   Position1 is before Position2, then -1 is returned. If Position1 is equal to Position2,
868   then 0 is returned. If Position1 is after Position2, then 1 is returned.
869 
870   @param  This                  A pointer to the EFI_S3_SAVE_STATE_PROTOCOL instance.
871   @param  Position1             The positions in the boot script table to compare
872   @param  Position2             The positions in the boot script table to compare
873   @param  RelativePosition      On return, points to the result of the comparison
874 
875   @retval EFI_SUCCESS           The operation succeeded.
876   @retval EFI_INVALID_PARAMETER The Position1 or Position2 is not a valid position in the boot script table.
877   @retval EFI_INVALID_PARAMETER The RelativePosition is NULL.
878 
879 **/
880 EFI_STATUS
881 EFIAPI
BootScriptCompare(IN CONST EFI_S3_SAVE_STATE_PROTOCOL * This,IN EFI_S3_BOOT_SCRIPT_POSITION Position1,IN EFI_S3_BOOT_SCRIPT_POSITION Position2,OUT UINTN * RelativePosition)882 BootScriptCompare (
883   IN CONST EFI_S3_SAVE_STATE_PROTOCOL      *This,
884   IN       EFI_S3_BOOT_SCRIPT_POSITION      Position1,
885   IN       EFI_S3_BOOT_SCRIPT_POSITION      Position2,
886   OUT      UINTN                           *RelativePosition
887   )
888 {
889 	return S3BootScriptCompare (Position1, Position2, RelativePosition);
890 }
891 /**
892   This routine is entry point of ScriptSave driver.
893 
894   @param  ImageHandle           Handle for this drivers loaded image protocol.
895   @param  SystemTable           EFI system table.
896 
897   @retval EFI_OUT_OF_RESOURCES  No enough resource
898   @retval EFI_SUCCESS           Succesfully installed the ScriptSave driver.
899   @retval other                 Errors occured.
900 
901 **/
902 EFI_STATUS
903 EFIAPI
InitializeS3SaveState(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)904 InitializeS3SaveState (
905   IN EFI_HANDLE           ImageHandle,
906   IN EFI_SYSTEM_TABLE     *SystemTable
907   )
908 {
909   EFI_STATUS    Status;
910   EFI_EVENT     EndOfDxeEvent;
911 
912   if (!PcdGetBool (PcdAcpiS3Enable)) {
913     return EFI_UNSUPPORTED;
914   }
915 
916   Status = gBS->CreateEventEx (
917                   EVT_NOTIFY_SIGNAL,
918                   TPL_CALLBACK,
919                   AcpiS3ContextSaveOnEndOfDxe,
920                   NULL,
921                   &gEfiEndOfDxeEventGroupGuid,
922                   &EndOfDxeEvent
923                   );
924   ASSERT_EFI_ERROR (Status);
925 
926   return  gBS->InstallProtocolInterface (
927                   &mHandle,
928                   &gEfiS3SaveStateProtocolGuid,
929                   EFI_NATIVE_INTERFACE,
930                   &mS3SaveState
931                   );
932 
933 }
934 
935 
936