Changeset 23498 in vbox
- Timestamp:
- Oct 2, 2009 6:14:07 AM (16 years ago)
- svn:sync-xref-src-repo-rev:
- 53130
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Devices/Network/DevVirtioNet.cpp
r23358 r23498 46 46 #define VIRTIO_RELOCATE(p, o) *(RTHCUINTPTR *)&p += o 47 47 48 /*****************************************************************************/ 49 RT_C_DECLS_BEGIN 50 PDMBOTHCBDECL(uint32_t) vnetGetHostFeatures(void *pState); 51 PDMBOTHCBDECL(int) vnetGetConfig(void *pState, uint32_t port, uint32_t cb, void *data); 52 PDMBOTHCBDECL(int) vnetSetConfig(void *pState, uint32_t port, uint32_t cb, void *data); 53 PDMBOTHCBDECL(void) vnetReset(void *pState); 54 RT_C_DECLS_END 55 56 /*****************************************************************************/ 48 #ifdef DEBUG 49 #define QUEUENAME(s, q) g_VPCIDevices[s->enmDevType].pfnGetQueueName(s, q) 50 #endif /* DEBUG */ 57 51 58 52 //- TODO: Move to Virtio.h ---------------------------------------------------- 53 54 #define VPCI_F_NOTIFY_ON_EMPTY 0x1000000 59 55 60 56 #define VRINGDESC_MAX_SIZE (2 * 1024 * 1024) … … 191 187 192 188 //- TODO: Move to VirtioPCI.cpp ----------------------------------------------- 189 190 /*****************************************************************************/ 191 RT_C_DECLS_BEGIN 192 PDMBOTHCBDECL(uint32_t) vnetGetHostFeatures(void *pState); 193 PDMBOTHCBDECL(int) vnetGetConfig(void *pState, uint32_t port, uint32_t cb, void *data); 194 PDMBOTHCBDECL(int) vnetSetConfig(void *pState, uint32_t port, uint32_t cb, void *data); 195 PDMBOTHCBDECL(void) vnetReset(void *pState); 196 #ifdef DEBUG 197 static const char *vnetGetQueueName(void *pvState, PVQUEUE pQueue); 198 #endif /* DEBUG */ 199 RT_C_DECLS_END 200 201 /*****************************************************************************/ 193 202 194 203 struct VirtioPCIDevices … … 206 215 int (*pfnSetConfig)(void *pvState, uint32_t port, uint32_t cb, void *data); 207 216 void (*pfnReset)(void *pvState); 217 #ifdef DEBUG 218 const char *(*pfnGetQueueName)(void *pvState, PVQUEUE pQueue); 219 #endif /* DEBUG */ 208 220 } g_VPCIDevices[] = 209 221 { 210 /* Vendor Device SSVendor SubSys Class Name */ 211 { 0x1AF4, 0x1000, 0x1AF4, 1 + VIRTIO_NET_ID, 0x0200, 3, "virtio-net", "vnet%d", 212 vnetGetHostFeatures, vnetGetConfig, vnetSetConfig, vnetReset }, /* Virtio Network Device */ 213 { 0x1AF4, 0x1001, 0x1AF4, 1 + VIRTIO_BLK_ID, 0x0180, 2, "virtio-blk", "vblk%d", 214 NULL, NULL, NULL, NULL }, /* Virtio Block Device */ 222 /* Vendor Device SSVendor SubSys Class NQ Name Instance */ 223 { /* Virtio Network Device */ 224 0x1AF4, 0x1000, 0x1AF4, 1 + VIRTIO_NET_ID, 0x0200, 3, "virtio-net", "vnet%d", 225 vnetGetHostFeatures, vnetGetConfig, vnetSetConfig, vnetReset 226 #ifdef DEBUG 227 , vnetGetQueueName 228 #endif /* DEBUG */ 229 }, 230 { /* Virtio Block Device */ 231 0x1AF4, 0x1001, 0x1AF4, 1 + VIRTIO_BLK_ID, 0x0180, 2, "virtio-blk", "vblk%d", 232 NULL, NULL, NULL, NULL 233 #ifdef DEBUG 234 , NULL 235 #endif /* DEBUG */ 236 }, 215 237 }; 216 238 … … 249 271 250 272 273 static void vqueueReset(PVQUEUE pQueue) 274 { 275 pQueue->VRing.addrDescriptors = 0; 276 pQueue->VRing.addrAvail = 0; 277 pQueue->VRing.addrUsed = 0; 278 pQueue->uNextAvailIndex = 0; 279 pQueue->uNextUsedIndex = 0; 280 } 281 251 282 static void vqueueInit(PVQUEUE pQueue, uint32_t uPageNumber) 252 283 { … … 287 318 void vringReadDesc(PVPCISTATE pState, PVRING pVRing, uint32_t uIndex, PVRINGDESC pDesc) 288 319 { 289 Log(("%s vringReadDesc: ring=%p idx=%u\n", INSTANCE(pState), pVRing, uIndex));320 //Log(("%s vringReadDesc: ring=%p idx=%u\n", INSTANCE(pState), pVRing, uIndex)); 290 321 PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), 291 322 pVRing->addrDescriptors + sizeof(VRINGDESC) * (uIndex % pVRing->uSize), … … 298 329 299 330 PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), 300 pVRing->addrAvail + RT_OFFSETOF(VRINGAVAIL, auRing[uIndex ]),331 pVRing->addrAvail + RT_OFFSETOF(VRINGAVAIL, auRing[uIndex % pVRing->uSize]), 301 332 &tmp, sizeof(tmp)); 302 333 return tmp; … … 310 341 pElem->nIn = pElem->nOut = 0; 311 342 312 Log2(("%s vqueueGet: avail_idx=%u\n", INSTANCE(pState), pQueue->uNextAvailIndex)); 343 Log2(("%s vqueueGet: %s avail_idx=%u\n", INSTANCE(pState), 344 QUEUENAME(pState, pQueue), pQueue->uNextAvailIndex)); 313 345 314 346 VRINGDESC desc; … … 322 354 if (desc.u16Flags & VRINGDESC_F_WRITE) 323 355 { 324 Log2(("%s vqueueGet: IN idx=%u seg=%u addr=%p cb=%u\n", INSTANCE(pState), idx,325 idx, desc.u64Addr, desc.uLen));356 Log2(("%s vqueueGet: %s IN seg=%u desc_idx=%u addr=%p cb=%u\n", INSTANCE(pState), 357 QUEUENAME(pState, pQueue), pElem->nIn, idx, desc.u64Addr, desc.uLen)); 326 358 pSeg = &pElem->aSegsIn[pElem->nIn++]; 327 359 } 328 360 else 329 361 { 330 Log2(("%s vqueueGet: OUT idx=%u seg=%u addr=%p cb=%u\n", INSTANCE(pState), idx,331 idx, desc.u64Addr, desc.uLen));362 Log2(("%s vqueueGet: %s OUT seg=%u desc_idx=%u addr=%p cb=%u\n", INSTANCE(pState), 363 QUEUENAME(pState, pQueue), pElem->nOut, idx, desc.u64Addr, desc.uLen)); 332 364 pSeg = &pElem->aSegsOut[pElem->nOut++]; 333 365 } … … 340 372 } while (desc.u16Flags & VRINGDESC_F_NEXT); 341 373 342 Log2(("%s vqueueGet: idx=%u nIn=%u nOut=%u\n", INSTANCE(pState),343 pElem->uIndex, pElem->nIn, pElem->nOut));374 Log2(("%s vqueueGet: %s head_desc_idx=%u nIn=%u nOut=%u\n", INSTANCE(pState), 375 QUEUENAME(pState, pQueue), pElem->uIndex, pElem->nIn, pElem->nOut)); 344 376 return true; 345 377 } … … 368 400 elem.uLen = uLen; 369 401 PDMDevHlpPhysWrite(pState->CTX_SUFF(pDevIns), 370 pVRing->addrUsed + RT_OFFSETOF(VRINGUSED, aRing[uIndex ]),402 pVRing->addrUsed + RT_OFFSETOF(VRINGUSED, aRing[uIndex % pVRing->uSize]), 371 403 &elem, sizeof(elem)); 372 404 } … … 376 408 unsigned int i, uOffset; 377 409 378 Log2(("%s vqueuePut: idx=%u acb=%u\n", INSTANCE(pState), pElem->uIndex, uLen)); 410 Log2(("%s vqueuePut: %s desc_idx=%u acb=%u\n", INSTANCE(pState), 411 QUEUENAME(pState, pQueue), pElem->uIndex, uLen)); 379 412 for (i = uOffset = 0; i < pElem->nIn && uOffset < uLen; i++) 380 413 { … … 382 415 if (pElem->aSegsIn[i].pv) 383 416 { 384 Log2(("%s vqueuePut: used_idx=%uidx=%u seg=%u addr=%p pv=%p cb=%u acb=%u\n", INSTANCE(pState),385 pQueue->uNextUsedIndex, pElem->uIndex, i, pElem->aSegsIn[i].addr, pElem->aSegsIn[i].pv, pElem->aSegsIn[i].cb, cbSegLen));417 Log2(("%s vqueuePut: %s used_idx=%u seg=%u addr=%p pv=%p cb=%u acb=%u\n", INSTANCE(pState), 418 QUEUENAME(pState, pQueue), pQueue->uNextUsedIndex, i, pElem->aSegsIn[i].addr, pElem->aSegsIn[i].pv, pElem->aSegsIn[i].cb, cbSegLen)); 386 419 PDMDevHlpPhysWrite(pState->CTX_SUFF(pDevIns), pElem->aSegsIn[i].addr, 387 420 pElem->aSegsIn[i].pv, cbSegLen); … … 390 423 } 391 424 392 vringWriteUsedElem(pState, &pQueue->VRing, pQueue->uNextUsedIndex, pElem->uIndex, uLen); 393 pQueue->uNextUsedIndex = (pQueue->uNextUsedIndex + 1) % pQueue->VRing.uSize; 425 Log2(("%s vqueuePut: %s used_idx=%u guest_used_idx=%u id=%u len=%u\n", INSTANCE(pState), 426 QUEUENAME(pState, pQueue), pQueue->uNextUsedIndex, vringReadUsedIndex(pState, &pQueue->VRing), pElem->uIndex, uLen)); 427 vringWriteUsedElem(pState, &pQueue->VRing, pQueue->uNextUsedIndex++, pElem->uIndex, uLen); 394 428 } 395 429 … … 403 437 void vqueueSync(PVPCISTATE pState, PVQUEUE pQueue) 404 438 { 405 Log2(("%s vqueueSync: used_idx=%u\n", INSTANCE(pState), pQueue->uNextUsedIndex)); 439 Log2(("%s vqueueSync: %s old_used_idx=%u new_used_idx=%u\n", INSTANCE(pState), 440 QUEUENAME(pState, pQueue), vringReadUsedIndex(pState, &pQueue->VRing), pQueue->uNextUsedIndex)); 406 441 vringWriteUsedIndex(pState, &pQueue->VRing, pQueue->uNextUsedIndex); 407 442 vqueueNotify(pState, pQueue); 443 } 444 445 void vpciReset(PVPCISTATE pState) 446 { 447 pState->uGuestFeatures = 0; 448 pState->uQueueSelector = 0; 449 pState->uStatus = 0; 450 pState->uISR = 0; 451 452 for (unsigned i = 0; i < g_VPCIDevices[pState->enmDevType].nQueues; i++) 453 vqueueReset(&pState->pQueues[i]); 408 454 } 409 455 … … 565 611 // TODO: Feature negotiation code goes here. 566 612 // The guest may potentially desire features we don't support! 613 pState->uGuestFeatures = u32; 567 614 break; 568 615 … … 988 1035 RTSEMEVENT hEventMoreRxDescAvail; 989 1036 990 R3PTRTYPE(PPDMQUEUE) pCanRxQueueR3; /**< Rx wakeup signaller - R3. */991 R0PTRTYPE(PPDMQUEUE) pCanRxQueueR0; /**< Rx wakeup signaller - R0. */992 RCPTRTYPE(PPDMQUEUE) pCanRxQueueRC; /**< Rx wakeup signaller - RC. */993 994 1037 /* Statistic fields ******************************************************/ 995 1038 … … 1032 1075 #define STATUS pState->config.uStatus 1033 1076 1077 #ifdef DEBUG 1078 static const char *vnetGetQueueName(void *pvState, PVQUEUE pQueue) 1079 { 1080 VNETSTATE *pState = (VNETSTATE *)pvState; 1081 if (pQueue == pState->pRxQueue) 1082 return "RX "; 1083 else if (pQueue == pState->pTxQueue) 1084 return "TX "; 1085 else if (pQueue == pState->pCtlQueue) 1086 return "CTL"; 1087 return "Invalid"; 1088 } 1089 #endif /* DEBUG */ 1090 1034 1091 PDMBOTHCBDECL(uint32_t) vnetGetHostFeatures(void *pvState) 1035 1092 { … … 1071 1128 VNETSTATE *pState = (VNETSTATE*)pvState; 1072 1129 Log(("%s Reset triggered\n", INSTANCE(pState))); 1130 vpciReset(&pState->VPCI); 1073 1131 // TODO: Implement reset 1074 1132 } … … 1133 1191 static int vnetCanReceive(VNETSTATE *pState) 1134 1192 { 1135 return (vqueueIsReady(&pState->VPCI, pState->pRxQueue) 1193 LogFlow(("%s vnetCanReceive\n", INSTANCE(pState))); 1194 int rc = (vqueueIsReady(&pState->VPCI, pState->pRxQueue) 1136 1195 && !vqueueIsEmpty(&pState->VPCI, pState->pRxQueue)) 1137 1196 ? VINF_SUCCESS : VERR_NET_NO_BUFFER_SPACE; 1197 LogFlow(("%s vnetCanReceive -> %d\n", INSTANCE(pState), rc)); 1198 return rc; 1138 1199 } 1139 1200 … … 1141 1202 { 1142 1203 VNETSTATE *pState = IFACE_TO_STATE(pInterface, INetworkPort); 1204 LogFlow(("%s vnetWaitReceiveAvail(cMillies=%u)\n", INSTANCE(pState), cMillies)); 1143 1205 int rc = vnetCanReceive(pState); 1144 1206 … … 1166 1228 ASMAtomicXchgBool(&pState->fMaybeOutOfSpace, false); 1167 1229 1230 LogFlow(("%s vnetWaitReceiveAvail -> %d\n", INSTANCE(pState), rc)); 1168 1231 return rc; 1169 1232 } … … 1225 1288 { 1226 1289 VNETHDR hdr; 1227 memset(&hdr, 0, sizeof(hdr)); 1228 /*hdr.u8Flags = 0;1229 hdr.u8GSOType = VNETHDR_GSO_NONE;*/1290 1291 hdr.u8Flags = 0; 1292 hdr.u8GSOType = VNETHDR_GSO_NONE; 1230 1293 1231 1294 unsigned int uOffset = 0; … … 1267 1330 uElemSize += uSize; 1268 1331 } 1269 elem.uIndex = nElem;1270 1332 vqueuePut(&pState->VPCI, pState->pRxQueue, &elem, uElemSize); 1271 1333 } … … 1367 1429 { 1368 1430 VNETSTATE *pState = (VNETSTATE*)pvState; 1369 Log(("%s Receive buffers has been added.\n", INSTANCE(pState))); 1431 Log(("%s Receive buffers has been added, waking up receive thread.\n", INSTANCE(pState))); 1432 vnetWakeupReceive(pState->VPCI.CTX_SUFF(pDevIns)); 1370 1433 } 1371 1434 … … 1420 1483 RTMemFree(pFrame); 1421 1484 } 1422 vqueuePut(&pState->VPCI, pQueue, &elem, uOffset);1485 vqueuePut(&pState->VPCI, pQueue, &elem, sizeof(VNETHDR) + uOffset); 1423 1486 vqueueSync(&pState->VPCI, pQueue); 1424 1487 } … … 1501 1564 return rc;*/ 1502 1565 1503 1504 /* Create the RX notifier signaller. */1505 rc = PDMDevHlpPDMQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 1, 0,1506 vnetCanRxQueueConsumer, true, "VNet-rcv", &pState->pCanRxQueueR3);1507 if (RT_FAILURE(rc))1508 return rc;1509 pState->pCanRxQueueR0 = PDMQueueR0Ptr(pState->pCanRxQueueR3);1510 pState->pCanRxQueueRC = PDMQueueRCPtr(pState->pCanRxQueueR3);1511 1566 1512 1567 /* Create Link Up Timer */
Note:
See TracChangeset
for help on using the changeset viewer.