Coverage Report

Created: 2026-04-28 06:34

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/hbfa-fl/HBFA/UefiHostFuzzTestCasePkg/TestStub/VirtioPciDevice10StubLib/VirtioPciDevice10StubLib.c
Line
Count
Source
1
/** @file
2
3
Copyright (c) 2018, Intel Corporation. All rights reserved.<BR>
4
SPDX-License-Identifier: BSD-2-Clause-Patent
5
6
**/
7
#include <Uefi.h>
8
#include <stdio.h>
9
#include <stdlib.h>
10
11
#include <Library/BaseLib.h>
12
#include <Library/DebugLib.h>
13
#include <Library/BaseMemoryLib.h>
14
#include <Library/MemoryAllocationLib.h>
15
#include <Protocol/BlockIo.h>
16
#include <Protocol/VirtioDevice.h>
17
#include <Protocol/PciIo.h>
18
#include <Protocol/PciRootBridgeIo.h>
19
#include <Library/VirtioPciDeviceStubLib.h>
20
#include <IndustryStandard/Virtio10.h>
21
#include <Library/PciCapPciIoLib.h>
22
#include <Library/PciCapLib.h>
23
24
PCI_CFG_SPACE *PciCfg;
25
26
// VOID
27
// EFIAPI
28
// PrintByByte(UINT8* Content, UINT32 Len) {
29
//   UINT32 i;
30
//   for (i = 0; i < Len; i++) {
31
//     if (i % 16 == 0) printf ("\n");
32
//     printf ("%02x ", Content[i]);
33
//   }
34
//   printf ("\n");
35
// }
36
37
EFI_STATUS
38
EFIAPI
39
PciCapPciIoDeviceInit (
40
  IN  EFI_PCI_IO_PROTOCOL *PciIo,
41
  OUT PCI_CAP_DEV         **PciDevice
42
  );
43
44
VOID
45
EFIAPI
46
PciCapPciIoDeviceUninit (
47
  IN PCI_CAP_DEV *PciDevice
48
  );
49
50
RETURN_STATUS
51
EFIAPI
52
PciCapListInit (
53
  IN  PCI_CAP_DEV  *PciDevice,
54
  OUT PCI_CAP_LIST **CapList
55
  );
56
57
RETURN_STATUS
58
EFIAPI
59
PciCapListFindCap (
60
  IN  PCI_CAP_LIST   *CapList,
61
  IN  PCI_CAP_DOMAIN Domain,
62
  IN  UINT16         CapId,
63
  IN  UINT16         Instance,
64
  OUT PCI_CAP        **Cap    OPTIONAL
65
  );
66
67
RETURN_STATUS
68
EFIAPI
69
PciCapRead (
70
  IN  PCI_CAP_DEV *PciDevice,
71
  IN  PCI_CAP     *Cap,
72
  IN  UINT16      SourceOffsetInCap,
73
  OUT VOID        *DestinationBuffer,
74
  IN  UINT16      Size
75
  );
76
77
VOID
78
EFIAPI
79
PciCapListUninit (
80
  IN PCI_CAP_LIST *CapList
81
  );
82
83
EFI_STATUS
84
GetBarType (
85
  IN  EFI_PCI_IO_PROTOCOL  *PciIo,
86
  IN  UINT8                BarIndex,
87
  OUT VIRTIO_1_0_BAR_TYPE  *BarType
88
  )
89
240
{
90
240
  if (BarIndex >= sizeof(PciCfg->PciBasicCfg.Device.Bar)/sizeof(UINT32)) {
91
18
    return EFI_INVALID_PARAMETER;
92
18
  }
93
94
222
  if (PciCfg->PciBasicCfg.Device.Bar[BarIndex] & BIT0) {
95
138
    *BarType = Virtio10BarTypeIo;
96
138
  }
97
84
  else {
98
84
    *BarType = Virtio10BarTypeMem;
99
84
  }
100
101
222
  return EFI_SUCCESS;
102
240
}
103
104
STATIC
105
EFI_STATUS
106
Virtio10Transfer (
107
  IN     EFI_PCI_IO_PROTOCOL  *PciIo,
108
  IN     VIRTIO_1_0_CONFIG    *Config,
109
  IN     BOOLEAN              Write,
110
  IN     UINTN                FieldOffset,
111
  IN     UINTN                FieldSize,
112
  IN OUT VOID                 *Buffer
113
  )
114
2.67k
{
115
2.67k
  if (!Config->Exists ||
116
2.67k
      (FieldSize > Config->Length) ||
117
2.67k
      (FieldOffset > Config->Length - FieldSize))
118
0
  {
119
0
    return EFI_INVALID_PARAMETER;
120
0
  }
121
  
122
2.67k
  if (Write) {
123
1.78k
    CopyMem ((void *) (((UINT64) PciCfg->PciBasicCfg.Device.Bar[1] << 32) | PciCfg->PciBasicCfg.Device.Bar[0] + Config->Offset + FieldOffset), Buffer, FieldSize);
124
1.78k
  }
125
893
  else {
126
893
    CopyMem (Buffer, (void *) (((UINT64) PciCfg->PciBasicCfg.Device.Bar[1] << 32) | PciCfg->PciBasicCfg.Device.Bar[0] + Config->Offset + FieldOffset), FieldSize);
127
893
  }
128
129
2.67k
  return EFI_SUCCESS;
130
2.67k
}
131
132
STATIC
133
EFI_STATUS
134
EFIAPI
135
Virtio10GetDeviceFeatures (
136
  IN VIRTIO_DEVICE_PROTOCOL  *This,
137
  OUT UINT64                 *DeviceFeatures
138
  )
139
161
{
140
161
  VIRTIO_1_0_DEV  *Dev;
141
161
  UINT32          Selector;
142
161
  UINT32          Features32[2];
143
144
161
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
145
146
483
  for (Selector = 0; Selector < 2; ++Selector) {
147
322
    EFI_STATUS  Status;
148
149
    //
150
    // Select the low or high half of the features.
151
    //
152
322
    Status = Virtio10Transfer (
153
322
               Dev->PciIo,
154
322
               &Dev->CommonConfig,
155
322
               TRUE,
156
322
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DeviceFeatureSelect),
157
322
               sizeof Selector,
158
322
               &Selector
159
322
               );
160
322
    if (EFI_ERROR (Status)) {
161
0
      return Status;
162
0
    }
163
164
    //
165
    // Fetch that half.
166
    //
167
322
    Status = Virtio10Transfer (
168
322
               Dev->PciIo,
169
322
               &Dev->CommonConfig,
170
322
               FALSE,
171
322
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DeviceFeature),
172
322
               sizeof Features32[Selector],
173
322
               &Features32[Selector]
174
322
               );
175
322
    if (EFI_ERROR (Status)) {
176
0
      return Status;
177
0
    }
178
322
  }
179
180
161
  *DeviceFeatures = LShiftU64 (Features32[1], 32) | Features32[0];
181
161
  return EFI_SUCCESS;
182
161
}
183
184
STATIC
185
EFI_STATUS
186
EFIAPI
187
Virtio10SetGuestFeatures (
188
  IN VIRTIO_DEVICE_PROTOCOL  *This,
189
  IN UINT64                  Features
190
  )
191
83
{
192
83
  VIRTIO_1_0_DEV  *Dev;
193
83
  UINT32          Selector;
194
83
  UINT32          Features32[2];
195
196
83
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
197
198
83
  Features32[0] = (UINT32)Features;
199
83
  Features32[1] = (UINT32)RShiftU64 (Features, 32);
200
201
249
  for (Selector = 0; Selector < 2; ++Selector) {
202
166
    EFI_STATUS  Status;
203
204
    //
205
    // Select the low or high half of the features.
206
    //
207
166
    Status = Virtio10Transfer (
208
166
               Dev->PciIo,
209
166
               &Dev->CommonConfig,
210
166
               TRUE,
211
166
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DriverFeatureSelect),
212
166
               sizeof Selector,
213
166
               &Selector
214
166
               );
215
166
    if (EFI_ERROR (Status)) {
216
0
      return Status;
217
0
    }
218
219
    //
220
    // Write that half.
221
    //
222
166
    Status = Virtio10Transfer (
223
166
               Dev->PciIo,
224
166
               &Dev->CommonConfig,
225
166
               TRUE,
226
166
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DriverFeature),
227
166
               sizeof Features32[Selector],
228
166
               &Features32[Selector]
229
166
               );
230
231
166
    if (EFI_ERROR (Status)) {
232
0
      return Status;
233
0
    }
234
166
  }
235
236
83
  return EFI_SUCCESS;
237
83
}
238
239
STATIC
240
EFI_STATUS
241
EFIAPI
242
Virtio10SetQueueAddress (
243
  IN VIRTIO_DEVICE_PROTOCOL  *This,
244
  IN VRING                   *Ring,
245
  IN UINT64                  RingBaseShift
246
  )
247
79
{
248
79
  VIRTIO_1_0_DEV  *Dev;
249
79
  EFI_STATUS      Status;
250
79
  UINT64          Address;
251
79
  UINT16          Enable;
252
253
79
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
254
255
79
  Address  = (UINTN)Ring->Desc;
256
79
  Address += RingBaseShift;
257
79
  Status   = Virtio10Transfer (
258
79
               Dev->PciIo,
259
79
               &Dev->CommonConfig,
260
79
               TRUE,
261
79
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueDesc),
262
79
               sizeof Address,
263
79
               &Address
264
79
               );
265
79
  if (EFI_ERROR (Status)) {
266
0
    return Status;
267
0
  }
268
269
79
  Address  = (UINTN)Ring->Avail.Flags;
270
79
  Address += RingBaseShift;
271
79
  Status   = Virtio10Transfer (
272
79
               Dev->PciIo,
273
79
               &Dev->CommonConfig,
274
79
               TRUE,
275
79
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueAvail),
276
79
               sizeof Address,
277
79
               &Address
278
79
               );
279
79
  if (EFI_ERROR (Status)) {
280
0
    return Status;
281
0
  }
282
283
79
  Address  = (UINTN)Ring->Used.Flags;
284
79
  Address += RingBaseShift;
285
79
  Status   = Virtio10Transfer (
286
79
               Dev->PciIo,
287
79
               &Dev->CommonConfig,
288
79
               TRUE,
289
79
               OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueUsed),
290
79
               sizeof Address,
291
79
               &Address
292
79
               );
293
79
  if (EFI_ERROR (Status)) {
294
0
    return Status;
295
0
  }
296
297
79
  Enable = 1;
298
79
  Status = Virtio10Transfer (
299
79
             Dev->PciIo,
300
79
             &Dev->CommonConfig,
301
79
             TRUE,
302
79
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueEnable),
303
79
             sizeof Enable,
304
79
             &Enable
305
79
             );
306
79
  return Status;
307
79
}
308
309
STATIC
310
EFI_STATUS
311
EFIAPI
312
Virtio10SetQueueSel (
313
  IN VIRTIO_DEVICE_PROTOCOL  *This,
314
  IN UINT16                  Index
315
  )
316
83
{
317
83
  VIRTIO_1_0_DEV  *Dev;
318
83
  EFI_STATUS      Status;
319
320
83
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
321
322
83
  Status = Virtio10Transfer (
323
83
             Dev->PciIo,
324
83
             &Dev->CommonConfig,
325
83
             TRUE,
326
83
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueSelect),
327
83
             sizeof Index,
328
83
             &Index
329
83
             );
330
83
  return Status;
331
83
}
332
333
STATIC
334
EFI_STATUS
335
EFIAPI
336
Virtio10SetQueueNotify (
337
  IN VIRTIO_DEVICE_PROTOCOL  *This,
338
  IN UINT16                  Index
339
  )
340
0
{
341
0
  VIRTIO_1_0_DEV  *Dev;
342
0
  EFI_STATUS      Status;
343
0
  UINT16          SavedQueueSelect;
344
0
  UINT16          NotifyOffset;
345
346
0
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
347
348
  //
349
  // Read NotifyOffset first. NotifyOffset is queue specific, so we have
350
  // to stash & restore the current queue selector around it.
351
  //
352
  // So, start with saving the current queue selector.
353
  //
354
0
  Status = Virtio10Transfer (
355
0
             Dev->PciIo,
356
0
             &Dev->CommonConfig,
357
0
             FALSE,
358
0
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueSelect),
359
0
             sizeof SavedQueueSelect,
360
0
             &SavedQueueSelect
361
0
             );
362
0
  if (EFI_ERROR (Status)) {
363
0
    return Status;
364
0
  }
365
366
  //
367
  // Select the requested queue.
368
  //
369
0
  Status = Virtio10Transfer (
370
0
             Dev->PciIo,
371
0
             &Dev->CommonConfig,
372
0
             TRUE,
373
0
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueSelect),
374
0
             sizeof Index,
375
0
             &Index
376
0
             );
377
0
  if (EFI_ERROR (Status)) {
378
0
    return Status;
379
0
  }
380
381
  //
382
  // Read the QueueNotifyOff field.
383
  //
384
0
  Status = Virtio10Transfer (
385
0
             Dev->PciIo,
386
0
             &Dev->CommonConfig,
387
0
             FALSE,
388
0
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueNotifyOff),
389
0
             sizeof NotifyOffset,
390
0
             &NotifyOffset
391
0
             );
392
0
  if (EFI_ERROR (Status)) {
393
0
    return Status;
394
0
  }
395
396
  //
397
  // Re-select the original queue.
398
  //
399
0
  Status = Virtio10Transfer (
400
0
             Dev->PciIo,
401
0
             &Dev->CommonConfig,
402
0
             TRUE,
403
0
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueSelect),
404
0
             sizeof SavedQueueSelect,
405
0
             &SavedQueueSelect
406
0
             );
407
0
  if (EFI_ERROR (Status)) {
408
0
    return Status;
409
0
  }
410
411
  //
412
  // We can now kick the queue.
413
  //
414
0
  Status = Virtio10Transfer (
415
0
             Dev->PciIo,
416
0
             &Dev->NotifyConfig,
417
0
             TRUE,
418
0
             (UINTN)NotifyOffset * Dev->NotifyOffsetMultiplier,
419
0
             sizeof Index,
420
0
             &Index
421
0
             );
422
423
0
  return Status;
424
0
}
425
426
STATIC
427
EFI_STATUS
428
EFIAPI
429
Virtio10SetQueueAlign (
430
  IN VIRTIO_DEVICE_PROTOCOL  *This,
431
  IN UINT32                  Alignment
432
  )
433
79
{
434
79
  return (Alignment == EFI_PAGE_SIZE) ? EFI_SUCCESS : EFI_UNSUPPORTED;
435
79
}
436
437
STATIC
438
EFI_STATUS
439
EFIAPI
440
Virtio10SetPageSize (
441
  IN VIRTIO_DEVICE_PROTOCOL  *This,
442
  IN UINT32                  PageSize
443
  )
444
161
{
445
161
  return (PageSize == EFI_PAGE_SIZE) ? EFI_SUCCESS : EFI_UNSUPPORTED;
446
161
}
447
448
STATIC
449
EFI_STATUS
450
EFIAPI
451
Virtio10GetQueueNumMax (
452
  IN  VIRTIO_DEVICE_PROTOCOL  *This,
453
  OUT UINT16                  *QueueNumMax
454
  )
455
162
{
456
162
  VIRTIO_1_0_DEV  *Dev;
457
162
  EFI_STATUS      Status;
458
459
162
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
460
461
162
  Status = Virtio10Transfer (
462
162
             Dev->PciIo,
463
162
             &Dev->CommonConfig,
464
162
             FALSE,
465
162
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, QueueSize),
466
162
             sizeof *QueueNumMax,
467
162
             QueueNumMax
468
162
             );
469
470
162
  return Status;
471
162
}
472
473
STATIC
474
EFI_STATUS
475
EFIAPI
476
Virtio10SetQueueNum (
477
  IN VIRTIO_DEVICE_PROTOCOL  *This,
478
  IN UINT16                  QueueSize
479
  )
480
79
{
481
79
  EFI_STATUS  Status;
482
79
  UINT16      CurrentSize;
483
484
  //
485
  // This member function is required for VirtIo MMIO, and a no-op in
486
  // VirtIo PCI 0.9.5. In VirtIo 1.0, drivers can theoretically use this
487
  // member to reduce memory consumption, but none of our drivers do. So
488
  // just check that they set the size that is already in effect.
489
  //
490
79
  Status = Virtio10GetQueueNumMax (This, &CurrentSize);
491
79
  if (EFI_ERROR (Status)) {
492
0
    return Status;
493
0
  }
494
495
79
  return (CurrentSize == QueueSize) ? EFI_SUCCESS : EFI_UNSUPPORTED;
496
79
}
497
498
STATIC
499
EFI_STATUS
500
EFIAPI
501
Virtio10GetDeviceStatus (
502
  IN  VIRTIO_DEVICE_PROTOCOL  *This,
503
  OUT UINT8                   *DeviceStatus
504
  )
505
83
{
506
83
  VIRTIO_1_0_DEV  *Dev;
507
83
  EFI_STATUS      Status;
508
  
509
83
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
510
  
511
83
  Status = Virtio10Transfer (
512
83
             Dev->PciIo,
513
83
             &Dev->CommonConfig,
514
83
             FALSE,
515
83
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DeviceStatus),
516
83
             sizeof *DeviceStatus,
517
83
             DeviceStatus
518
83
             );
519
83
  return Status;
520
83
}
521
522
STATIC
523
EFI_STATUS
524
EFIAPI
525
Virtio10SetDeviceStatus (
526
  IN VIRTIO_DEVICE_PROTOCOL  *This,
527
  IN UINT8                   DeviceStatus
528
  )
529
727
{
530
727
  VIRTIO_1_0_DEV  *Dev;
531
727
  EFI_STATUS      Status;
532
533
727
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
534
535
727
  Status = Virtio10Transfer (
536
727
             Dev->PciIo,
537
727
             &Dev->CommonConfig,
538
727
             TRUE,
539
727
             OFFSET_OF (VIRTIO_PCI_COMMON_CFG, DeviceStatus),
540
727
             sizeof DeviceStatus,
541
727
             &DeviceStatus
542
727
             );
543
727
  return Status;
544
727
}
545
546
STATIC
547
EFI_STATUS
548
EFIAPI
549
Virtio10WriteDevice (
550
  IN VIRTIO_DEVICE_PROTOCOL  *This,
551
  IN UINTN                   FieldOffset,
552
  IN UINTN                   FieldSize,
553
  IN UINT64                  Value
554
  )
555
0
{
556
0
  VIRTIO_1_0_DEV  *Dev;
557
0
  EFI_STATUS      Status;
558
559
0
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
560
561
0
  Status = Virtio10Transfer (
562
0
             Dev->PciIo,
563
0
             &Dev->SpecificConfig,
564
0
             TRUE,
565
0
             FieldOffset,
566
0
             FieldSize,
567
0
             &Value
568
0
             );
569
0
  return Status;
570
0
}
571
572
STATIC
573
EFI_STATUS
574
EFIAPI
575
Virtio10ReadDevice (
576
  IN  VIRTIO_DEVICE_PROTOCOL  *This,
577
  IN  UINTN                   FieldOffset,
578
  IN  UINTN                   FieldSize,
579
  IN  UINTN                   BufferSize,
580
  OUT VOID                    *Buffer
581
  )
582
326
{
583
326
  VIRTIO_1_0_DEV  *Dev;
584
326
  EFI_STATUS      Status;
585
586
326
  if (FieldSize != BufferSize) {
587
0
    return EFI_INVALID_PARAMETER;
588
0
  }
589
590
326
  Dev = VIRTIO_1_0_FROM_VIRTIO_DEVICE (This);
591
592
326
  Status = Virtio10Transfer (
593
326
             Dev->PciIo,
594
326
             &Dev->SpecificConfig,
595
326
             FALSE,
596
326
             FieldOffset,
597
326
             FieldSize,
598
326
             Buffer
599
326
             );
600
326
  return Status;
601
326
}
602
603
STATIC
604
EFI_STATUS
605
EFIAPI
606
Virtio10AllocateSharedPages (
607
  IN     VIRTIO_DEVICE_PROTOCOL  *This,
608
  IN     UINTN                   Pages,
609
  IN OUT VOID                    **HostAddress
610
  )
611
79
{
612
79
  EFI_STATUS      Status = EFI_SUCCESS;
613
  
614
79
  *HostAddress = AllocatePages (Pages);
615
616
79
  if (*HostAddress == NULL) {
617
0
    Status = EFI_OUT_OF_RESOURCES;
618
0
  }
619
620
79
  return Status;
621
79
}
622
623
STATIC
624
VOID
625
EFIAPI
626
Virtio10FreeSharedPages (
627
  IN  VIRTIO_DEVICE_PROTOCOL  *This,
628
  IN  UINTN                   Pages,
629
  IN  VOID                    *HostAddress
630
  )
631
0
{
632
0
  FreePages(HostAddress, Pages);
633
0
}
634
635
STATIC
636
EFI_STATUS
637
EFIAPI
638
Virtio10MapSharedBuffer (
639
  IN     VIRTIO_DEVICE_PROTOCOL  *This,
640
  IN     VIRTIO_MAP_OPERATION    Operation,
641
  IN     VOID                    *HostAddress,
642
  IN OUT UINTN                   *NumberOfBytes,
643
  OUT    EFI_PHYSICAL_ADDRESS    *DeviceAddress,
644
  OUT    VOID                    **Mapping
645
  )
646
79
{
647
79
  return EFI_SUCCESS;
648
79
}
649
650
STATIC
651
EFI_STATUS
652
EFIAPI
653
Virtio10UnmapSharedBuffer (
654
  IN  VIRTIO_DEVICE_PROTOCOL  *This,
655
  IN  VOID                    *Mapping
656
  )
657
0
{
658
0
  return EFI_SUCCESS;
659
0
}
660
661
STATIC CONST VIRTIO_DEVICE_PROTOCOL  mVirtIoTemplate = {
662
  VIRTIO_SPEC_REVISION (1,     0, 0),
663
  0,                           // SubSystemDeviceId, filled in dynamically
664
  Virtio10GetDeviceFeatures,
665
  Virtio10SetGuestFeatures,
666
  Virtio10SetQueueAddress,
667
  Virtio10SetQueueSel,
668
  Virtio10SetQueueNotify,
669
  Virtio10SetQueueAlign,
670
  Virtio10SetPageSize,
671
  Virtio10GetQueueNumMax,
672
  Virtio10SetQueueNum,
673
  Virtio10GetDeviceStatus,
674
  Virtio10SetDeviceStatus,
675
  Virtio10WriteDevice,
676
  Virtio10ReadDevice,
677
  Virtio10AllocateSharedPages,
678
  Virtio10FreeSharedPages,
679
  Virtio10MapSharedBuffer,
680
  Virtio10UnmapSharedBuffer
681
};
682
683
EFI_STATUS
684
EFIAPI
685
FixPciCfg (
686
  IN OUT PCI_CFG_SPACE             *PciCfg,
687
  IN VIRTIO_PCI_CAP_COMMON_CONFIG  *PciCommonConfig
688
)
689
0
{ 
690
0
  PciCfg->PciBasicCfg.Device.Bar[0] = (UINT32) ((UINT64)PciCommonConfig);
691
 
692
0
  return EFI_SUCCESS;
693
0
}
694
695
STATIC
696
EFI_STATUS
697
ParseCapabilities (
698
  IN OUT VIRTIO_1_0_DEV *Device
699
  )
700
396
{
701
396
  EFI_STATUS   Status;
702
396
  PCI_CAP_DEV  *PciDevice;
703
396
  PCI_CAP_LIST *CapList;
704
396
  UINT16       VendorInstance;
705
396
  PCI_CAP      *VendorCap;
706
707
396
  Status = PciCapPciIoDeviceInit (Device->PciIo, &PciDevice);
708
396
  if (EFI_ERROR (Status)) {
709
0
    return Status;
710
0
  }
711
712
396
  Status = PciCapListInit (PciDevice, &CapList);
713
396
  if (EFI_ERROR (Status)) {
714
115
    goto UninitPciDevice;
715
115
  }
716
717
281
  for (VendorInstance = 0;
718
665
       !EFI_ERROR (PciCapListFindCap (CapList, PciCapNormal,
719
281
                     EFI_PCI_CAPABILITY_ID_VENDOR, VendorInstance,
720
281
                     &VendorCap));
721
426
       VendorInstance++) {
722
426
    UINT8             CapLen;
723
426
    VIRTIO_PCI_CAP    VirtIoCap;
724
426
    VIRTIO_1_0_CONFIG *ParsedConfig;
725
726
    //
727
    // Big enough to accommodate a VIRTIO_PCI_CAP structure?
728
    //
729
426
    Status = PciCapRead (PciDevice, VendorCap,
730
426
               OFFSET_OF (EFI_PCI_CAPABILITY_VENDOR_HDR, Length), &CapLen,
731
426
               sizeof CapLen);
732
426
    if (EFI_ERROR (Status)) {
733
0
      goto UninitCapList;
734
0
    }
735
426
    if (CapLen < sizeof VirtIoCap) {
736
      //
737
      // Too small, move to next.
738
      //
739
243
      continue;
740
243
    }
741
742
    //
743
    // Read interesting part of capability.
744
    //
745
183
    Status = PciCapRead (PciDevice, VendorCap, 0, &VirtIoCap, sizeof VirtIoCap);
746
183
    if (EFI_ERROR (Status)) {
747
28
      goto UninitCapList;
748
28
    }
749
750
155
    switch (VirtIoCap.ConfigType) {
751
27
    case VIRTIO_PCI_CAP_COMMON_CFG:
752
27
      ParsedConfig = &Device->CommonConfig;
753
27
      break;
754
71
    case VIRTIO_PCI_CAP_NOTIFY_CFG:
755
71
      ParsedConfig = &Device->NotifyConfig;
756
71
      break;
757
22
    case VIRTIO_PCI_CAP_DEVICE_CFG:
758
22
      ParsedConfig = &Device->SpecificConfig;
759
22
      break;
760
35
    default:
761
      //
762
      // Capability is not interesting.
763
      //
764
35
      continue;
765
155
    }
766
767
    //
768
    // Save the location of the register block into ParsedConfig.
769
    //
770
120
    Status = GetBarType (Device->PciIo, VirtIoCap.Bar, &ParsedConfig->BarType);
771
120
    if (EFI_ERROR (Status)) {
772
9
      goto UninitCapList;
773
9
    }
774
111
    ParsedConfig->Bar    = VirtIoCap.Bar;
775
111
    ParsedConfig->Offset = VirtIoCap.Offset;
776
111
    ParsedConfig->Length = VirtIoCap.Length;
777
778
111
    if (VirtIoCap.ConfigType == VIRTIO_PCI_CAP_NOTIFY_CFG) {
779
      //
780
      // This capability has an additional field called NotifyOffsetMultiplier;
781
      // parse it too.
782
      //
783
66
      if (CapLen < sizeof VirtIoCap + sizeof Device->NotifyOffsetMultiplier) {
784
        //
785
        // Too small, move to next.
786
        //
787
24
        continue;
788
24
      }
789
790
42
      Status = PciCapRead (PciDevice, VendorCap, sizeof VirtIoCap,
791
42
                 &Device->NotifyOffsetMultiplier,
792
42
                 sizeof Device->NotifyOffsetMultiplier);
793
42
      if (EFI_ERROR (Status)) {
794
5
        goto UninitCapList;
795
5
      }
796
42
    }
797
798
    //
799
    // Capability parsed successfully.
800
    //
801
82
    ParsedConfig->Exists = TRUE;
802
82
  }
803
804
239
  ASSERT_EFI_ERROR (Status);
805
806
281
UninitCapList:
807
281
  PciCapListUninit (CapList);
808
809
396
UninitPciDevice:
810
396
  PciCapPciIoDeviceUninit (PciDevice);
811
812
396
  return Status;
813
281
}
814
815
EFI_STATUS
816
EFIAPI
817
PciIoRead (
818
  IN EFI_PCI_IO_PROTOCOL       *PciIo,
819
  IN EFI_PCI_IO_PROTOCOL_WIDTH Width,
820
  IN UINT32                    Offset,
821
  IN UINTN                     Count,
822
  IN OUT VOID                  *Buffer
823
21.4k
) {
824
21.4k
  UINT16 Len = 0;
825
826
21.4k
  switch (Width)
827
21.4k
  {
828
6.97k
  case EfiPciIoWidthUint32:
829
6.97k
    Len = Count * sizeof(UINT32);
830
6.97k
    break;
831
12.8k
  case EfiPciIoWidthUint16:
832
12.8k
    Len = Count * sizeof(UINT16);
833
12.8k
    break;
834
1.69k
  case EfiPciIoWidthUint8:
835
1.69k
    Len = Count * sizeof(UINT8);
836
1.69k
    break;
837
0
  default:
838
0
    break;
839
21.4k
  }
840
841
21.4k
  CopyMem (Buffer, (void *) ((UINT64) PciCfg + Offset), Len);
842
21.4k
  return EFI_SUCCESS;
843
21.4k
}
844
845
EFI_STATUS
846
EFIAPI
847
InitVirtioPciDevice (
848
  IN OUT  VIRTIO_1_0_DEV         *Device,
849
  IN      UINT8                  *TestBuffer,
850
  IN      UINTN                  BufferSize,
851
  IN      EFI_PCI_IO_PROTOCOL    *PciIo
852
) 
853
499
{
854
  // VOID  *ConfigRegion;
855
  
856
  // ConfigRegion = (VOID *) AllocateZeroPool (sizeof (PCI_CFG_SPACE));
857
499
  PciCfg = (PCI_CFG_SPACE *) TestBuffer; 
858
859
  // CopyMem (PciCfg, (void *) TestBuffer, sizeof (PCI_CFG_SPACE));
860
  
861
499
  PciIo->Pci.Read = &PciIoRead;
862
863
499
  if ((PciCfg->PciBasicCfg.Hdr.VendorId == VIRTIO_VENDOR_ID) &&
864
465
      (PciCfg->PciBasicCfg.Hdr.DeviceId >= 0x1040) &&
865
444
      (PciCfg->PciBasicCfg.Hdr.DeviceId <= 0x107F) &&
866
422
      (PciCfg->PciBasicCfg.Hdr.RevisionID >= 0x01) &&
867
415
      (PciCfg->PciBasicCfg.Device.SubsystemID >= 0x40) &&
868
399
      ((PciCfg->PciBasicCfg.Hdr.Status & EFI_PCI_STATUS_CAPABILITY) != 0))
869
396
  {
870
396
    if (!((PciCfg->PciBasicCfg.Hdr.DeviceId != 0x1050) || !(PciCfg->PciBasicCfg.Hdr.ClassCode == PciCfg))) {
871
0
      return EFI_OUT_OF_RESOURCES;
872
0
    }
873
396
  }
874
103
  else {
875
103
      return EFI_OUT_OF_RESOURCES;
876
103
  }
877
878
396
  ParseCapabilities (Device);
879
880
396
  return EFI_SUCCESS;
881
499
}
882
883
884
EFI_STATUS
885
EFIAPI
886
ParseBufferAndInitVirtioPciDev10 (
887
  IN      UINT8                   *TestBuffer,
888
  IN      UINTN                   BufferSize,
889
  IN      VOID                    *ConfigRegion,
890
  IN OUT  VIRTIO_1_0_DEV          *Device
891
) 
892
298
{
893
298
  EFI_STATUS                    Status;
894
298
  EFI_PCI_IO_PROTOCOL           *PciIo;
895
298
  VIRTIO_BLK_CONFIG             *BlkConfig;
896
298
  VIRTIO_PCI_CAP_COMMON_CONFIG  *PciCommonConfig;
897
298
  VOID                          *PciNotifyConfig;
898
  // VOID                          *ConfigRegion;
899
900
  // ConfigRegion = (VOID *) AllocatePool(sizeof (PCI_CFG_SPACE) + sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG) + sizeof (VIRTIO_BLK_CONFIG) + 0x100);
901
298
  if (ConfigRegion == NULL) {
902
0
    return EFI_OUT_OF_RESOURCES;
903
0
  }
904
905
298
  if (BufferSize != sizeof (PCI_CFG_SPACE) + sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG) + sizeof (VIRTIO_BLK_CONFIG)) {
906
66
    goto FreeDevice;
907
66
  }
908
909
232
  Device->Signature = VIRTIO_1_0_SIGNATURE;
910
232
  CopyMem (&Device->VirtIo, &mVirtIoTemplate, sizeof (VIRTIO_DEVICE_PROTOCOL));
911
912
232
  PciCfg = (PCI_CFG_SPACE *) ConfigRegion;
913
232
  PciCommonConfig = (VIRTIO_PCI_CAP_COMMON_CONFIG *) ((UINT64) ConfigRegion + sizeof (PCI_CFG_SPACE));
914
232
  BlkConfig = (VIRTIO_BLK_CONFIG *) ((UINT64) ConfigRegion + sizeof (PCI_CFG_SPACE) + sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG));
915
232
  PciNotifyConfig = (VOID *) ((UINT64) ConfigRegion + sizeof (PCI_CFG_SPACE) + sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG) + sizeof (VIRTIO_BLK_CONFIG));
916
917
232
  CopyMem (PciCfg, (void *) TestBuffer, sizeof (PCI_CFG_SPACE));
918
  
919
232
  CopyMem (PciCommonConfig, 
920
232
           (void *) ((UINT64)TestBuffer + sizeof (PCI_CFG_SPACE)), 
921
232
           sizeof (VIRTIO_PCI_CAP_COMMON_CONFIG));
922
923
232
  CopyMem (BlkConfig,
924
232
           (void *) ((UINT64)TestBuffer + sizeof (PCI_CFG_SPACE) + sizeof (VIRTIO_PCI_CAP_COMMON_CONFIG)), 
925
232
           sizeof (VIRTIO_BLK_CONFIG));
926
927
232
  if ((PciCfg->PciBasicCfg.Hdr.VendorId == VIRTIO_VENDOR_ID) &&
928
209
      (PciCfg->PciBasicCfg.Hdr.DeviceId >= 0x1040) &&
929
194
      (PciCfg->PciBasicCfg.Hdr.DeviceId <= 0x107F) &&
930
178
      (PciCfg->PciBasicCfg.Hdr.RevisionID >= 0x01) &&
931
177
      (PciCfg->PciBasicCfg.Device.SubsystemID >= 0x40) &&
932
170
      ((PciCfg->PciBasicCfg.Hdr.Status & EFI_PCI_STATUS_CAPABILITY) != 0))
933
169
  {
934
169
    if (!((PciCfg->PciBasicCfg.Hdr.DeviceId != 0x1050) || !(PciCfg->PciBasicCfg.Hdr.ClassCode == PciCfg))) {
935
0
      goto FreeDevice;
936
0
    }
937
169
  }
938
63
  else {
939
63
      goto FreeDevice;
940
63
  }
941
942
169
  PciCfg->PciBasicCfg.Device.Bar[0] = (UINT32) ((UINT64)PciCommonConfig);
943
169
  PciCfg->PciBasicCfg.Device.Bar[1] = (UINT32) ((UINT64)PciCommonConfig >> 32);
944
  
945
169
  Device->VirtIo.SubSystemDeviceId = PciCfg->PciBasicCfg.Hdr.DeviceId - 0x1040;
946
947
169
  Device->CommonConfig.Offset   = 0;
948
169
  Device->CommonConfig.Bar      = 0;
949
169
  Device->CommonConfig.Exists   = TRUE;
950
169
  Device->CommonConfig.Length   = sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG);
951
  
952
169
  Device->NotifyConfig.Offset   = sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG) + sizeof (VIRTIO_BLK_CONFIG);
953
169
  Device->NotifyConfig.Bar      = 0;
954
169
  Device->NotifyConfig.Exists   = TRUE;
955
169
  Device->NotifyConfig.Length   = 0x100;
956
957
169
  Device->SpecificConfig.Offset = sizeof(VIRTIO_PCI_CAP_COMMON_CONFIG);
958
169
  Device->SpecificConfig.Bar    = 0;
959
169
  Device->SpecificConfig.Exists = TRUE;
960
169
  Device->SpecificConfig.Length = sizeof (VIRTIO_BLK_CONFIG);
961
962
  // if (Device->NotifyConfig.Length < (Device->CommonConfig.Offset * Device->NotifyOffsetMultiplier + 4)) {
963
  //   goto FreeDevice;
964
  // }
965
966
169
  return EFI_SUCCESS;
967
968
129
FreeDevice:
969
  // FreePool (ConfigRegion);
970
971
129
  return EFI_OUT_OF_RESOURCES;
972
232
}