• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2   This file provides some helper functions which are specific for SD card device.
3 
4   Copyright (c) 2015 - 2016, Intel Corporation. All rights reserved.<BR>
5   This program and the accompanying materials
6   are licensed and made available under the terms and conditions of the BSD License
7   which accompanies this distribution.  The full text of the license may be found at
8   http://opensource.org/licenses/bsd-license.php
9 
10   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
11   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
12 
13 **/
14 
15 #include "SdMmcPciHcDxe.h"
16 
17 /**
18   Send command GO_IDLE_STATE to the device to make it go to Idle State.
19 
20   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
21 
22   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
23   @param[in] Slot           The slot number of the SD card to send the command to.
24 
25   @retval EFI_SUCCESS       The SD device is reset correctly.
26   @retval Others            The device reset fails.
27 
28 **/
29 EFI_STATUS
SdCardReset(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot)30 SdCardReset (
31   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
32   IN UINT8                              Slot
33   )
34 {
35   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
36   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
37   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
38   EFI_STATUS                            Status;
39 
40   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
41   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
42   ZeroMem (&Packet, sizeof (Packet));
43 
44   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
45   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
46   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
47 
48   SdMmcCmdBlk.CommandIndex = SD_GO_IDLE_STATE;
49   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBc;
50 
51   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
52 
53   return Status;
54 }
55 
56 /**
57   Send command SEND_IF_COND to the device to inquiry the SD Memory Card interface
58   condition.
59 
60   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
61 
62   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
63   @param[in] Slot           The slot number of the SD card to send the command to.
64   @param[in] SupplyVoltage  The supplied voltage by the host.
65   @param[in] CheckPattern   The check pattern to be sent to the device.
66 
67   @retval EFI_SUCCESS       The operation is done correctly.
68   @retval Others            The operation fails.
69 
70 **/
71 EFI_STATUS
SdCardVoltageCheck(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT8 SupplyVoltage,IN UINT8 CheckPattern)72 SdCardVoltageCheck (
73   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
74   IN UINT8                              Slot,
75   IN UINT8                              SupplyVoltage,
76   IN UINT8                              CheckPattern
77   )
78 {
79   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
80   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
81   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
82   EFI_STATUS                            Status;
83 
84   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
85   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
86   ZeroMem (&Packet, sizeof (Packet));
87 
88   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
89   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
90   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
91 
92   SdMmcCmdBlk.CommandIndex = SD_SEND_IF_COND;
93   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBcr;
94   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR7;
95   SdMmcCmdBlk.CommandArgument = (SupplyVoltage << 8) | CheckPattern;
96 
97   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
98 
99   if (!EFI_ERROR (Status)) {
100     if (SdMmcStatusBlk.Resp0 != SdMmcCmdBlk.CommandArgument) {
101       return EFI_DEVICE_ERROR;
102     }
103   }
104 
105   return Status;
106 }
107 
108 /**
109   Send command SDIO_SEND_OP_COND to the device to see whether it is SDIO device.
110 
111   Refer to SDIO Simplified Spec 3 Section 3.2 for details.
112 
113   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
114   @param[in] Slot           The slot number of the SD card to send the command to.
115   @param[in] VoltageWindow  The supply voltage window.
116   @param[in] S18R           The boolean to show if it should switch to 1.8v.
117 
118   @retval EFI_SUCCESS       The operation is done correctly.
119   @retval Others            The operation fails.
120 
121 **/
122 EFI_STATUS
SdioSendOpCond(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT32 VoltageWindow,IN BOOLEAN S18R)123 SdioSendOpCond (
124   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
125   IN UINT8                              Slot,
126   IN UINT32                             VoltageWindow,
127   IN BOOLEAN                            S18R
128   )
129 {
130   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
131   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
132   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
133   EFI_STATUS                            Status;
134   UINT32                                Switch;
135 
136   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
137   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
138   ZeroMem (&Packet, sizeof (Packet));
139 
140   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
141   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
142   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
143 
144   SdMmcCmdBlk.CommandIndex = SDIO_SEND_OP_COND;
145   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBcr;
146   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR4;
147 
148   Switch = S18R ? BIT24 : 0;
149 
150   SdMmcCmdBlk.CommandArgument = (VoltageWindow & 0xFFFFFF) | Switch;
151 
152   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
153 
154   return Status;
155 }
156 
157 /**
158   Send command SD_SEND_OP_COND to the device to see whether it is SDIO device.
159 
160   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
161 
162   @param[in]  PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
163   @param[in]  Slot           The slot number of the SD card to send the command to.
164   @param[in]  Rca            The relative device address of addressed device.
165   @param[in]  VoltageWindow  The supply voltage window.
166   @param[in]  S18R           The boolean to show if it should switch to 1.8v.
167   @param[in]  Xpc            The boolean to show if it should provide 0.36w power control.
168   @param[in]  Hcs            The boolean to show if it support host capacity info.
169   @param[out] Ocr            The buffer to store returned OCR register value.
170 
171   @retval EFI_SUCCESS        The operation is done correctly.
172   @retval Others             The operation fails.
173 
174 **/
175 EFI_STATUS
SdCardSendOpCond(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,IN UINT32 VoltageWindow,IN BOOLEAN S18R,IN BOOLEAN Xpc,IN BOOLEAN Hcs,OUT UINT32 * Ocr)176 SdCardSendOpCond (
177   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
178   IN     UINT8                          Slot,
179   IN     UINT16                         Rca,
180   IN     UINT32                         VoltageWindow,
181   IN     BOOLEAN                        S18R,
182   IN     BOOLEAN                        Xpc,
183   IN     BOOLEAN                        Hcs,
184      OUT UINT32                         *Ocr
185   )
186 {
187   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
188   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
189   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
190   EFI_STATUS                            Status;
191   UINT32                                Switch;
192   UINT32                                MaxPower;
193   UINT32                                HostCapacity;
194 
195   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
196   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
197   ZeroMem (&Packet, sizeof (Packet));
198 
199   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
200   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
201   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
202 
203   SdMmcCmdBlk.CommandIndex = SD_APP_CMD;
204   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
205   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
206   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
207 
208   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
209   if (EFI_ERROR (Status)) {
210     return Status;
211   }
212 
213   SdMmcCmdBlk.CommandIndex = SD_SEND_OP_COND;
214   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBcr;
215   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR3;
216 
217   Switch       = S18R ? BIT24 : 0;
218   MaxPower     = Xpc ? BIT28 : 0;
219   HostCapacity = Hcs ? BIT30 : 0;
220 
221   SdMmcCmdBlk.CommandArgument = (VoltageWindow & 0xFFFFFF) | Switch | MaxPower | HostCapacity;
222 
223   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
224   if (!EFI_ERROR (Status)) {
225     //
226     // For details, refer to SD Host Controller Simplified Spec 3.0 Table 2-12.
227     //
228     *Ocr = SdMmcStatusBlk.Resp0;
229   }
230 
231   return Status;
232 }
233 
234 /**
235   Broadcast command ALL_SEND_CID to the bus to ask all the SD devices to send the
236   data of their CID registers.
237 
238   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
239 
240   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
241   @param[in] Slot           The slot number of the SD card to send the command to.
242 
243   @retval EFI_SUCCESS       The operation is done correctly.
244   @retval Others            The operation fails.
245 
246 **/
247 EFI_STATUS
SdCardAllSendCid(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot)248 SdCardAllSendCid (
249   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
250   IN UINT8                              Slot
251   )
252 {
253   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
254   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
255   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
256   EFI_STATUS                            Status;
257 
258   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
259   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
260   ZeroMem (&Packet, sizeof (Packet));
261 
262   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
263   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
264   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
265 
266   SdMmcCmdBlk.CommandIndex = SD_ALL_SEND_CID;
267   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBcr;
268   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR2;
269 
270   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
271 
272   return Status;
273 }
274 
275 /**
276   Send command SET_RELATIVE_ADDR to the SD device to assign a Relative device
277   Address (RCA).
278 
279   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
280 
281   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
282   @param[in]  Slot          The slot number of the SD card to send the command to.
283   @param[out] Rca           The relative device address to assign.
284 
285   @retval EFI_SUCCESS       The operation is done correctly.
286   @retval Others            The operation fails.
287 
288 **/
289 EFI_STATUS
SdCardSetRca(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,OUT UINT16 * Rca)290 SdCardSetRca (
291   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
292   IN     UINT8                          Slot,
293      OUT UINT16                         *Rca
294   )
295 {
296   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
297   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
298   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
299   EFI_STATUS                            Status;
300 
301   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
302   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
303   ZeroMem (&Packet, sizeof (Packet));
304 
305   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
306   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
307   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
308 
309   SdMmcCmdBlk.CommandIndex = SD_SET_RELATIVE_ADDR;
310   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeBcr;
311   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR6;
312 
313   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
314   if (!EFI_ERROR (Status)) {
315     *Rca = (UINT16)(SdMmcStatusBlk.Resp0 >> 16);
316   }
317 
318   return Status;
319 }
320 
321 /**
322   Send command SEND_CSD to the SD device to get the data of the CSD register.
323 
324   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
325 
326   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
327   @param[in]  Slot          The slot number of the SD card to send the command to.
328   @param[in]  Rca           The relative device address of selected device.
329   @param[out] Csd           The buffer to store the content of the CSD register.
330                             Note the caller should ignore the lowest byte of this
331                             buffer as the content of this byte is meaningless even
332                             if the operation succeeds.
333 
334   @retval EFI_SUCCESS       The operation is done correctly.
335   @retval Others            The operation fails.
336 
337 **/
338 EFI_STATUS
SdCardGetCsd(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,OUT SD_CSD * Csd)339 SdCardGetCsd (
340   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
341   IN     UINT8                          Slot,
342   IN     UINT16                         Rca,
343      OUT SD_CSD                         *Csd
344   )
345 {
346   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
347   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
348   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
349   EFI_STATUS                            Status;
350 
351   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
352   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
353   ZeroMem (&Packet, sizeof (Packet));
354 
355   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
356   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
357   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
358 
359   SdMmcCmdBlk.CommandIndex = SD_SEND_CSD;
360   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
361   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR2;
362   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
363 
364   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
365   if (!EFI_ERROR (Status)) {
366     //
367     // For details, refer to SD Host Controller Simplified Spec 3.0 Table 2-12.
368     //
369     CopyMem (((UINT8*)Csd) + 1, &SdMmcStatusBlk.Resp0, sizeof (SD_CSD) - 1);
370   }
371 
372   return Status;
373 }
374 
375 /**
376   Send command SEND_CSD to the SD device to get the data of the CSD register.
377 
378   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
379 
380   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
381   @param[in]  Slot          The slot number of the SD card to send the command to.
382   @param[in]  Rca           The relative device address of selected device.
383   @param[out] Scr           The buffer to store the content of the SCR register.
384 
385   @retval EFI_SUCCESS       The operation is done correctly.
386   @retval Others            The operation fails.
387 
388 **/
389 EFI_STATUS
SdCardGetScr(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,OUT SD_SCR * Scr)390 SdCardGetScr (
391   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
392   IN     UINT8                          Slot,
393   IN     UINT16                         Rca,
394      OUT SD_SCR                         *Scr
395   )
396 {
397   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
398   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
399   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
400   EFI_STATUS                            Status;
401 
402   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
403   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
404   ZeroMem (&Packet, sizeof (Packet));
405 
406   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
407   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
408   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
409 
410   SdMmcCmdBlk.CommandIndex = SD_APP_CMD;
411   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
412   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
413   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
414 
415   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
416   if (EFI_ERROR (Status)) {
417     return Status;
418   }
419 
420   SdMmcCmdBlk.CommandIndex = SD_SEND_SCR;
421   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAdtc;
422   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
423 
424   Packet.InDataBuffer     = Scr;
425   Packet.InTransferLength = sizeof (SD_SCR);
426 
427   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
428 
429   return Status;
430 }
431 
432 /**
433   Send command SELECT_DESELECT_CARD to the SD device to select/deselect it.
434 
435   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
436 
437   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
438   @param[in]  Slot          The slot number of the SD card to send the command to.
439   @param[in]  Rca           The relative device address of selected device.
440 
441   @retval EFI_SUCCESS       The operation is done correctly.
442   @retval Others            The operation fails.
443 
444 **/
445 EFI_STATUS
SdCardSelect(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca)446 SdCardSelect (
447   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
448   IN UINT8                              Slot,
449   IN UINT16                             Rca
450   )
451 {
452   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
453   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
454   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
455   EFI_STATUS                            Status;
456 
457   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
458   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
459   ZeroMem (&Packet, sizeof (Packet));
460 
461   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
462   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
463   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
464 
465   SdMmcCmdBlk.CommandIndex = SD_SELECT_DESELECT_CARD;
466   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
467   if (Rca != 0) {
468     SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1b;
469   }
470   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
471 
472   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
473 
474   return Status;
475 }
476 
477 /**
478   Send command VOLTAGE_SWITCH to the SD device to switch the voltage of the device.
479 
480   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
481 
482   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
483   @param[in]  Slot          The slot number of the SD card to send the command to.
484 
485   @retval EFI_SUCCESS       The operation is done correctly.
486   @retval Others            The operation fails.
487 
488 **/
489 EFI_STATUS
SdCardVoltageSwitch(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot)490 SdCardVoltageSwitch (
491   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
492   IN UINT8                              Slot
493   )
494 {
495   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
496   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
497   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
498   EFI_STATUS                            Status;
499 
500   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
501   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
502   ZeroMem (&Packet, sizeof (Packet));
503 
504   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
505   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
506   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
507 
508   SdMmcCmdBlk.CommandIndex = SD_VOLTAGE_SWITCH;
509   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
510   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
511   SdMmcCmdBlk.CommandArgument = 0;
512 
513   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
514 
515   return Status;
516 }
517 
518 /**
519   Send command SET_BUS_WIDTH to the SD device to set the bus width.
520 
521   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
522 
523   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
524   @param[in] Slot           The slot number of the SD card to send the command to.
525   @param[in] Rca            The relative device address of addressed device.
526   @param[in] BusWidth       The bus width to be set, it could be 1 or 4.
527 
528   @retval EFI_SUCCESS       The operation is done correctly.
529   @retval Others            The operation fails.
530 
531 **/
532 EFI_STATUS
SdCardSetBusWidth(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,IN UINT8 BusWidth)533 SdCardSetBusWidth (
534   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
535   IN UINT8                              Slot,
536   IN UINT16                             Rca,
537   IN UINT8                              BusWidth
538   )
539 {
540   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
541   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
542   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
543   EFI_STATUS                            Status;
544   UINT8                                 Value;
545 
546   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
547   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
548   ZeroMem (&Packet, sizeof (Packet));
549 
550   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
551   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
552   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
553 
554   SdMmcCmdBlk.CommandIndex = SD_APP_CMD;
555   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
556   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
557   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
558 
559   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
560   if (EFI_ERROR (Status)) {
561     return Status;
562   }
563 
564   SdMmcCmdBlk.CommandIndex = SD_SET_BUS_WIDTH;
565   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
566   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
567 
568   if (BusWidth == 1) {
569     Value = 0;
570   } else if (BusWidth == 4) {
571     Value = 2;
572   } else {
573     return EFI_INVALID_PARAMETER;
574   }
575 
576   SdMmcCmdBlk.CommandArgument = Value & 0x3;
577 
578   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
579   return Status;
580 }
581 
582 /**
583   Send command SWITCH_FUNC to the SD device to check switchable function or switch card function.
584 
585   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
586 
587   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
588   @param[in]  Slot          The slot number of the SD card to send the command to.
589   @param[in]  AccessMode    The value for access mode group.
590   @param[in]  CommandSystem The value for command set group.
591   @param[in]  DriveStrength The value for drive length group.
592   @param[in]  PowerLimit    The value for power limit group.
593   @param[in]  Mode          Switch or check function.
594   @param[out] SwitchResp    The return switch function status.
595 
596   @retval EFI_SUCCESS       The operation is done correctly.
597   @retval Others            The operation fails.
598 
599 **/
600 EFI_STATUS
SdCardSwitch(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT8 AccessMode,IN UINT8 CommandSystem,IN UINT8 DriveStrength,IN UINT8 PowerLimit,IN BOOLEAN Mode,OUT UINT8 * SwitchResp)601 SdCardSwitch (
602   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
603   IN     UINT8                          Slot,
604   IN     UINT8                          AccessMode,
605   IN     UINT8                          CommandSystem,
606   IN     UINT8                          DriveStrength,
607   IN     UINT8                          PowerLimit,
608   IN     BOOLEAN                        Mode,
609      OUT UINT8                          *SwitchResp
610   )
611 {
612   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
613   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
614   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
615   EFI_STATUS                            Status;
616   UINT32                                ModeValue;
617 
618   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
619   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
620   ZeroMem (&Packet, sizeof (Packet));
621 
622   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
623   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
624   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
625 
626   SdMmcCmdBlk.CommandIndex = SD_SWITCH_FUNC;
627   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAdtc;
628   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
629 
630   ModeValue = Mode ? BIT31 : 0;
631   SdMmcCmdBlk.CommandArgument = (AccessMode & 0xF) | ((PowerLimit & 0xF) << 4) | \
632                                 ((DriveStrength & 0xF) << 8) | ((DriveStrength & 0xF) << 12) | \
633                                 ModeValue;
634 
635   Packet.InDataBuffer     = SwitchResp;
636   Packet.InTransferLength = 64;
637 
638   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
639 
640   return Status;
641 }
642 
643 /**
644   Send command SEND_STATUS to the addressed SD device to get its status register.
645 
646   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
647 
648   @param[in]  PassThru      A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
649   @param[in]  Slot          The slot number of the SD card to send the command to.
650   @param[in]  Rca           The relative device address of addressed device.
651   @param[out] DevStatus     The returned device status.
652 
653   @retval EFI_SUCCESS       The operation is done correctly.
654   @retval Others            The operation fails.
655 
656 **/
657 EFI_STATUS
SdCardSendStatus(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,OUT UINT32 * DevStatus)658 SdCardSendStatus (
659   IN     EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru,
660   IN     UINT8                          Slot,
661   IN     UINT16                         Rca,
662      OUT UINT32                         *DevStatus
663   )
664 {
665   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
666   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
667   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
668   EFI_STATUS                            Status;
669 
670   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
671   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
672   ZeroMem (&Packet, sizeof (Packet));
673 
674   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
675   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
676   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
677 
678   SdMmcCmdBlk.CommandIndex = SD_SEND_STATUS;
679   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAc;
680   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
681   SdMmcCmdBlk.CommandArgument = (UINT32)Rca << 16;
682 
683   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
684   if (!EFI_ERROR (Status)) {
685     *DevStatus = SdMmcStatusBlk.Resp0;
686   }
687 
688   return Status;
689 }
690 
691 /**
692   Send command SEND_TUNING_BLOCK to the SD device for HS200 optimal sampling point
693   detection.
694 
695   It may be sent up to 40 times until the host finishes the tuning procedure.
696 
697   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 for details.
698 
699   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
700   @param[in] Slot           The slot number of the SD card to send the command to.
701 
702   @retval EFI_SUCCESS       The operation is done correctly.
703   @retval Others            The operation fails.
704 
705 **/
706 EFI_STATUS
SdCardSendTuningBlk(IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot)707 SdCardSendTuningBlk (
708   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
709   IN UINT8                              Slot
710   )
711 {
712   EFI_SD_MMC_COMMAND_BLOCK              SdMmcCmdBlk;
713   EFI_SD_MMC_STATUS_BLOCK               SdMmcStatusBlk;
714   EFI_SD_MMC_PASS_THRU_COMMAND_PACKET   Packet;
715   EFI_STATUS                            Status;
716   UINT8                                 TuningBlock[64];
717 
718   ZeroMem (&SdMmcCmdBlk, sizeof (SdMmcCmdBlk));
719   ZeroMem (&SdMmcStatusBlk, sizeof (SdMmcStatusBlk));
720   ZeroMem (&Packet, sizeof (Packet));
721 
722   Packet.SdMmcCmdBlk    = &SdMmcCmdBlk;
723   Packet.SdMmcStatusBlk = &SdMmcStatusBlk;
724   Packet.Timeout        = SD_MMC_HC_GENERIC_TIMEOUT;
725 
726   SdMmcCmdBlk.CommandIndex = SD_SEND_TUNING_BLOCK;
727   SdMmcCmdBlk.CommandType  = SdMmcCommandTypeAdtc;
728   SdMmcCmdBlk.ResponseType = SdMmcResponseTypeR1;
729   SdMmcCmdBlk.CommandArgument = 0;
730 
731   Packet.InDataBuffer     = TuningBlock;
732   Packet.InTransferLength = sizeof (TuningBlock);
733 
734   Status = SdMmcPassThruPassThru (PassThru, Slot, &Packet, NULL);
735 
736   return Status;
737 }
738 
739 /**
740   Tunning the sampling point of SDR104 or SDR50 bus speed mode.
741 
742   Command SD_SEND_TUNING_BLOCK may be sent up to 40 times until the host finishes the
743   tuning procedure.
744 
745   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 and
746   SD Host Controller Simplified Spec 3.0 section Figure 3-7 for details.
747 
748   @param[in] PciIo          A pointer to the EFI_PCI_IO_PROTOCOL instance.
749   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
750   @param[in] Slot           The slot number of the SD card to send the command to.
751 
752   @retval EFI_SUCCESS       The operation is done correctly.
753   @retval Others            The operation fails.
754 
755 **/
756 EFI_STATUS
SdCardTuningClock(IN EFI_PCI_IO_PROTOCOL * PciIo,IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot)757 SdCardTuningClock (
758   IN EFI_PCI_IO_PROTOCOL                *PciIo,
759   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
760   IN UINT8                              Slot
761   )
762 {
763   EFI_STATUS          Status;
764   UINT8               HostCtrl2;
765   UINT8               Retry;
766 
767   //
768   // Notify the host that the sampling clock tuning procedure starts.
769   //
770   HostCtrl2 = BIT6;
771   Status = SdMmcHcOrMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, sizeof (HostCtrl2), &HostCtrl2);
772   if (EFI_ERROR (Status)) {
773     return Status;
774   }
775   //
776   // Ask the device to send a sequence of tuning blocks till the tuning procedure is done.
777   //
778   Retry = 0;
779   do {
780     Status = SdCardSendTuningBlk (PassThru, Slot);
781     if (EFI_ERROR (Status)) {
782       DEBUG ((DEBUG_ERROR, "SdCardSendTuningBlk: Send tuning block fails with %r\n", Status));
783       return Status;
784     }
785 
786     Status = SdMmcHcRwMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, TRUE, sizeof (HostCtrl2), &HostCtrl2);
787     if (EFI_ERROR (Status)) {
788       return Status;
789     }
790 
791     if ((HostCtrl2 & (BIT6 | BIT7)) == 0) {
792       break;
793     }
794     if ((HostCtrl2 & (BIT6 | BIT7)) == BIT7) {
795       return EFI_SUCCESS;
796     }
797   } while (++Retry < 40);
798 
799   DEBUG ((DEBUG_ERROR, "SdCardTuningClock: Send tuning block fails at %d times with HostCtrl2 %02x\n", Retry, HostCtrl2));
800   //
801   // Abort the tuning procedure and reset the tuning circuit.
802   //
803   HostCtrl2 = (UINT8)~(BIT6 | BIT7);
804   Status = SdMmcHcAndMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, sizeof (HostCtrl2), &HostCtrl2);
805   if (EFI_ERROR (Status)) {
806     return Status;
807   }
808   return EFI_DEVICE_ERROR;
809 }
810 
811 /**
812   Switch the bus width to specified width.
813 
814   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 and
815   SD Host Controller Simplified Spec 3.0 section Figure 3-7 for details.
816 
817   @param[in] PciIo          A pointer to the EFI_PCI_IO_PROTOCOL instance.
818   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
819   @param[in] Slot           The slot number of the SD card to send the command to.
820   @param[in] Rca            The relative device address to be assigned.
821   @param[in] BusWidth       The bus width to be set, it could be 4 or 8.
822 
823   @retval EFI_SUCCESS       The operation is done correctly.
824   @retval Others            The operation fails.
825 
826 **/
827 EFI_STATUS
SdCardSwitchBusWidth(IN EFI_PCI_IO_PROTOCOL * PciIo,IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,IN UINT8 BusWidth)828 SdCardSwitchBusWidth (
829   IN EFI_PCI_IO_PROTOCOL                *PciIo,
830   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
831   IN UINT8                              Slot,
832   IN UINT16                             Rca,
833   IN UINT8                              BusWidth
834   )
835 {
836   EFI_STATUS          Status;
837   UINT32              DevStatus;
838 
839   Status = SdCardSetBusWidth (PassThru, Slot, Rca, BusWidth);
840   if (EFI_ERROR (Status)) {
841     DEBUG ((DEBUG_ERROR, "SdCardSwitchBusWidth: Switch to bus width %d fails with %r\n", BusWidth, Status));
842     return Status;
843   }
844 
845   Status = SdCardSendStatus (PassThru, Slot, Rca, &DevStatus);
846   if (EFI_ERROR (Status)) {
847     DEBUG ((DEBUG_ERROR, "SdCardSwitchBusWidth: Send status fails with %r\n", Status));
848     return Status;
849   }
850   //
851   // Check the switch operation is really successful or not.
852   //
853   if ((DevStatus >> 16) != 0) {
854     DEBUG ((DEBUG_ERROR, "SdCardSwitchBusWidth: The switch operation fails as DevStatus is 0x%08x\n", DevStatus));
855     return EFI_DEVICE_ERROR;
856   }
857 
858   Status = SdMmcHcSetBusWidth (PciIo, Slot, BusWidth);
859 
860   return Status;
861 }
862 
863 /**
864   Switch the high speed timing according to request.
865 
866   Refer to SD Physical Layer Simplified Spec 4.1 Section 4.7 and
867   SD Host Controller Simplified Spec 3.0 section Figure 2-29 for details.
868 
869   @param[in] PciIo          A pointer to the EFI_PCI_IO_PROTOCOL instance.
870   @param[in] PassThru       A pointer to the EFI_SD_MMC_PASS_THRU_PROTOCOL instance.
871   @param[in] Slot           The slot number of the SD card to send the command to.
872   @param[in] Rca            The relative device address to be assigned.
873   @param[in] S18A           The boolean to show if it's a UHS-I SD card.
874 
875   @retval EFI_SUCCESS       The operation is done correctly.
876   @retval Others            The operation fails.
877 
878 **/
879 EFI_STATUS
SdCardSetBusMode(IN EFI_PCI_IO_PROTOCOL * PciIo,IN EFI_SD_MMC_PASS_THRU_PROTOCOL * PassThru,IN UINT8 Slot,IN UINT16 Rca,IN BOOLEAN S18A)880 SdCardSetBusMode (
881   IN EFI_PCI_IO_PROTOCOL                *PciIo,
882   IN EFI_SD_MMC_PASS_THRU_PROTOCOL      *PassThru,
883   IN UINT8                              Slot,
884   IN UINT16                             Rca,
885   IN BOOLEAN                            S18A
886   )
887 {
888   EFI_STATUS                   Status;
889   SD_MMC_HC_SLOT_CAP           *Capability;
890   UINT32                       ClockFreq;
891   UINT8                        BusWidth;
892   UINT8                        AccessMode;
893   UINT8                        HostCtrl1;
894   UINT8                        HostCtrl2;
895   UINT8                        SwitchResp[64];
896   SD_MMC_HC_PRIVATE_DATA       *Private;
897 
898   Private = SD_MMC_HC_PRIVATE_FROM_THIS (PassThru);
899 
900   Capability = &Private->Capability[Slot];
901 
902   Status = SdCardSelect (PassThru, Slot, Rca);
903   if (EFI_ERROR (Status)) {
904     return Status;
905   }
906 
907   BusWidth = 4;
908 
909   Status = SdCardSwitchBusWidth (PciIo, PassThru, Slot, Rca, BusWidth);
910   if (EFI_ERROR (Status)) {
911     return Status;
912   }
913   //
914   // Get the supported bus speed from SWITCH cmd return data group #1.
915   //
916   Status = SdCardSwitch (PassThru, Slot, 0xF, 0xF, 0xF, 0xF, FALSE, SwitchResp);
917   if (EFI_ERROR (Status)) {
918     return Status;
919   }
920   //
921   // Calculate supported bus speed/bus width/clock frequency by host and device capability.
922   //
923   ClockFreq = 0;
924   if (S18A && (Capability->Sdr104 != 0) && ((SwitchResp[13] & BIT3) != 0)) {
925     ClockFreq = 208;
926     AccessMode = 3;
927   } else if (S18A && (Capability->Sdr50 != 0) && ((SwitchResp[13] & BIT2) != 0)) {
928     ClockFreq = 100;
929     AccessMode = 2;
930   } else if (S18A && (Capability->Ddr50 != 0) && ((SwitchResp[13] & BIT4) != 0)) {
931     ClockFreq = 50;
932     AccessMode = 4;
933   } else if ((SwitchResp[13] & BIT1) != 0) {
934     ClockFreq = 50;
935     AccessMode = 1;
936   } else {
937     ClockFreq = 25;
938     AccessMode = 0;
939   }
940 
941   Status = SdCardSwitch (PassThru, Slot, AccessMode, 0xF, 0xF, 0xF, TRUE, SwitchResp);
942   if (EFI_ERROR (Status)) {
943     return Status;
944   }
945 
946   if ((SwitchResp[16] & 0xF) != AccessMode) {
947     DEBUG ((DEBUG_ERROR, "SdCardSetBusMode: Switch to AccessMode %d ClockFreq %d BusWidth %d fails! The Switch response is 0x%1x\n", AccessMode, ClockFreq, BusWidth, SwitchResp[16] & 0xF));
948     return EFI_DEVICE_ERROR;
949   }
950 
951   DEBUG ((DEBUG_INFO, "SdCardSetBusMode: Switch to AccessMode %d ClockFreq %d BusWidth %d\n", AccessMode, ClockFreq, BusWidth));
952 
953   //
954   // Set to Hight Speed timing
955   //
956   if (AccessMode == 1) {
957     HostCtrl1 = BIT2;
958     Status = SdMmcHcOrMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL1, sizeof (HostCtrl1), &HostCtrl1);
959     if (EFI_ERROR (Status)) {
960       return Status;
961     }
962   }
963 
964   HostCtrl2 = (UINT8)~0x7;
965   Status = SdMmcHcAndMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, sizeof (HostCtrl2), &HostCtrl2);
966   if (EFI_ERROR (Status)) {
967     return Status;
968   }
969   HostCtrl2 = AccessMode;
970   Status = SdMmcHcOrMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, sizeof (HostCtrl2), &HostCtrl2);
971   if (EFI_ERROR (Status)) {
972     return Status;
973   }
974 
975   Status = SdMmcHcClockSupply (PciIo, Slot, ClockFreq * 1000, *Capability);
976   if (EFI_ERROR (Status)) {
977     return Status;
978   }
979 
980   if ((AccessMode == 3) || ((AccessMode == 2) && (Capability->TuningSDR50 != 0))) {
981     Status = SdCardTuningClock (PciIo, PassThru, Slot);
982     if (EFI_ERROR (Status)) {
983       return Status;
984     }
985   }
986 
987   return Status;
988 }
989 
990 /**
991   Execute SD device identification procedure.
992 
993   Refer to SD Physical Layer Simplified Spec 4.1 Section 3.6 for details.
994 
995   @param[in] Private        A pointer to the SD_MMC_HC_PRIVATE_DATA instance.
996   @param[in] Slot           The slot number of the SD card to send the command to.
997 
998   @retval EFI_SUCCESS       There is a SD card.
999   @retval Others            There is not a SD card.
1000 
1001 **/
1002 EFI_STATUS
SdCardIdentification(IN SD_MMC_HC_PRIVATE_DATA * Private,IN UINT8 Slot)1003 SdCardIdentification (
1004   IN SD_MMC_HC_PRIVATE_DATA             *Private,
1005   IN UINT8                              Slot
1006   )
1007 {
1008   EFI_STATUS                     Status;
1009   EFI_PCI_IO_PROTOCOL            *PciIo;
1010   EFI_SD_MMC_PASS_THRU_PROTOCOL  *PassThru;
1011   UINT32                         Ocr;
1012   UINT16                         Rca;
1013   BOOLEAN                        Xpc;
1014   BOOLEAN                        S18r;
1015   UINT64                         MaxCurrent;
1016   UINT16                         ControllerVer;
1017   UINT8                          PowerCtrl;
1018   UINT32                         PresentState;
1019   UINT8                          HostCtrl2;
1020 
1021   PciIo    = Private->PciIo;
1022   PassThru = &Private->PassThru;
1023   //
1024   // 1. Send Cmd0 to the device
1025   //
1026   Status = SdCardReset (PassThru, Slot);
1027   if (EFI_ERROR (Status)) {
1028     DEBUG ((DEBUG_INFO, "SdCardIdentification: Executing Cmd0 fails with %r\n", Status));
1029     return Status;
1030   }
1031   //
1032   // 2. Send Cmd8 to the device
1033   //
1034   Status = SdCardVoltageCheck (PassThru, Slot, 0x1, 0xFF);
1035   if (EFI_ERROR (Status)) {
1036     DEBUG ((DEBUG_INFO, "SdCardIdentification: Executing Cmd8 fails with %r\n", Status));
1037     return Status;
1038   }
1039   //
1040   // 3. Send SDIO Cmd5 to the device to the SDIO device OCR register.
1041   //
1042   Status = SdioSendOpCond (PassThru, Slot, 0, FALSE);
1043   if (!EFI_ERROR (Status)) {
1044     DEBUG ((DEBUG_INFO, "SdCardIdentification: Found SDIO device, ignore it as we don't support\n"));
1045     return EFI_DEVICE_ERROR;
1046   }
1047   //
1048   // 4. Send Acmd41 with voltage window 0 to the device
1049   //
1050   Status = SdCardSendOpCond (PassThru, Slot, 0, 0, FALSE, FALSE, FALSE, &Ocr);
1051   if (EFI_ERROR (Status)) {
1052     DEBUG ((DEBUG_INFO, "SdCardIdentification: Executing SdCardSendOpCond fails with %r\n", Status));
1053     return EFI_DEVICE_ERROR;
1054   }
1055 
1056   if (Private->Capability[Slot].Voltage33 != 0) {
1057     //
1058     // Support 3.3V
1059     //
1060     MaxCurrent = ((UINT32)Private->MaxCurrent[Slot] & 0xFF) * 4;
1061   } else if (Private->Capability[Slot].Voltage30 != 0) {
1062     //
1063     // Support 3.0V
1064     //
1065     MaxCurrent = (((UINT32)Private->MaxCurrent[Slot] >> 8) & 0xFF) * 4;
1066   } else if (Private->Capability[Slot].Voltage18 != 0) {
1067     //
1068     // Support 1.8V
1069     //
1070     MaxCurrent = (((UINT32)Private->MaxCurrent[Slot] >> 16) & 0xFF) * 4;
1071   } else {
1072     ASSERT (FALSE);
1073     return EFI_DEVICE_ERROR;
1074   }
1075 
1076   if (MaxCurrent >= 150) {
1077     Xpc = TRUE;
1078   } else {
1079     Xpc = FALSE;
1080   }
1081 
1082   Status = SdMmcHcRwMmio (PciIo, Slot, SD_MMC_HC_CTRL_VER, TRUE, sizeof (ControllerVer), &ControllerVer);
1083   if (EFI_ERROR (Status)) {
1084     return Status;
1085   }
1086 
1087   if ((ControllerVer & 0xFF) == 2) {
1088     S18r = TRUE;
1089   } else if (((ControllerVer & 0xFF) == 0) || ((ControllerVer & 0xFF) == 1)) {
1090     S18r = FALSE;
1091   } else {
1092     ASSERT (FALSE);
1093     return EFI_UNSUPPORTED;
1094   }
1095   //
1096   // 5. Repeatly send Acmd41 with supply voltage window to the device.
1097   //    Note here we only support the cards complied with SD physical
1098   //    layer simplified spec version 2.0 and version 3.0 and above.
1099   //
1100   do {
1101     Status = SdCardSendOpCond (PassThru, Slot, 0, Ocr, S18r, Xpc, TRUE, &Ocr);
1102     if (EFI_ERROR (Status)) {
1103       DEBUG ((DEBUG_ERROR, "SdCardIdentification: SdCardSendOpCond fails with %r Ocr %x, S18r %x, Xpc %x\n", Status, Ocr, S18r, Xpc));
1104       return EFI_DEVICE_ERROR;
1105     }
1106   } while ((Ocr & BIT31) == 0);
1107 
1108   //
1109   // 6. If the S18A bit is set and the Host Controller supports 1.8V signaling
1110   //    (One of support bits is set to 1: SDR50, SDR104 or DDR50 in the
1111   //    Capabilities register), switch its voltage to 1.8V.
1112   //
1113   if ((Private->Capability[Slot].Sdr50 != 0 ||
1114        Private->Capability[Slot].Sdr104 != 0 ||
1115        Private->Capability[Slot].Ddr50 != 0) &&
1116        ((Ocr & BIT24) != 0)) {
1117     Status = SdCardVoltageSwitch (PassThru, Slot);
1118     if (EFI_ERROR (Status)) {
1119       DEBUG ((DEBUG_ERROR, "SdCardIdentification: Executing SdCardVoltageSwitch fails with %r\n", Status));
1120       Status = EFI_DEVICE_ERROR;
1121       goto Error;
1122     } else {
1123       Status = SdMmcHcStopClock (PciIo, Slot);
1124       if (EFI_ERROR (Status)) {
1125         Status = EFI_DEVICE_ERROR;
1126         goto Error;
1127       }
1128 
1129       SdMmcHcRwMmio (PciIo, Slot, SD_MMC_HC_PRESENT_STATE, TRUE, sizeof (PresentState), &PresentState);
1130       if (((PresentState >> 20) & 0xF) != 0) {
1131         DEBUG ((DEBUG_ERROR, "SdCardIdentification: SwitchVoltage fails with PresentState = 0x%x\n", PresentState));
1132         Status = EFI_DEVICE_ERROR;
1133         goto Error;
1134       }
1135       HostCtrl2  = BIT3;
1136       SdMmcHcOrMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, sizeof (HostCtrl2), &HostCtrl2);
1137 
1138       gBS->Stall (5000);
1139 
1140       SdMmcHcRwMmio (PciIo, Slot, SD_MMC_HC_HOST_CTRL2, TRUE, sizeof (HostCtrl2), &HostCtrl2);
1141       if ((HostCtrl2 & BIT3) == 0) {
1142         DEBUG ((DEBUG_ERROR, "SdCardIdentification: SwitchVoltage fails with HostCtrl2 = 0x%x\n", HostCtrl2));
1143         Status = EFI_DEVICE_ERROR;
1144         goto Error;
1145       }
1146 
1147       SdMmcHcInitClockFreq (PciIo, Slot, Private->Capability[Slot]);
1148 
1149       gBS->Stall (1000);
1150 
1151       SdMmcHcRwMmio (PciIo, Slot, SD_MMC_HC_PRESENT_STATE, TRUE, sizeof (PresentState), &PresentState);
1152       if (((PresentState >> 20) & 0xF) != 0xF) {
1153         DEBUG ((DEBUG_ERROR, "SdCardIdentification: SwitchVoltage fails with PresentState = 0x%x, It should be 0xF\n", PresentState));
1154         Status = EFI_DEVICE_ERROR;
1155         goto Error;
1156       }
1157     }
1158     DEBUG ((DEBUG_INFO, "SdCardIdentification: Switch to 1.8v signal voltage success\n"));
1159   }
1160 
1161   Status = SdCardAllSendCid (PassThru, Slot);
1162   if (EFI_ERROR (Status)) {
1163     DEBUG ((DEBUG_ERROR, "SdCardIdentification: Executing SdCardAllSendCid fails with %r\n", Status));
1164     return Status;
1165   }
1166 
1167   Status = SdCardSetRca (PassThru, Slot, &Rca);
1168   if (EFI_ERROR (Status)) {
1169     DEBUG ((DEBUG_ERROR, "SdCardIdentification: Executing SdCardSetRca fails with %r\n", Status));
1170     return Status;
1171   }
1172   //
1173   // Enter Data Tranfer Mode.
1174   //
1175   DEBUG ((DEBUG_INFO, "SdCardIdentification: Found a SD device at slot [%d]\n", Slot));
1176   Private->Slot[Slot].CardType = SdCardType;
1177 
1178   Status = SdCardSetBusMode (PciIo, PassThru, Slot, Rca, ((Ocr & BIT24) != 0));
1179 
1180   return Status;
1181 
1182 Error:
1183   //
1184   // Set SD Bus Power = 0
1185   //
1186   PowerCtrl = (UINT8)~BIT0;
1187   Status = SdMmcHcAndMmio (PciIo, Slot, SD_MMC_HC_POWER_CTRL, sizeof (PowerCtrl), &PowerCtrl);
1188   return EFI_DEVICE_ERROR;
1189 }
1190 
1191