Changeset 41660 in vbox
- Timestamp:
- Jun 12, 2012 8:08:17 AM (13 years ago)
- Location:
- trunk/src/VBox/Runtime/r0drv/linux
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Runtime/r0drv/linux/memobj-r0drv-linux.c
r41116 r41660 101 101 { 102 102 /** @todo fix rtR0ProcessToLinuxTask!! */ 103 /** @todo many (all?) callers currently assume that we return 'current'! */ 103 104 return R0Process == RTR0ProcHandleSelf() ? current : NULL; 104 105 } … … 166 167 return fKernel ? MY_PAGE_KERNEL_EXEC : PAGE_SHARED_EXEC; 167 168 } 169 } 170 171 172 /** 173 * Worker for rtR0MemObjNativeReserveUser and rtR0MemObjNativerMapUser that creates 174 * an empty user space mapping. 175 * 176 * We acquire the mmap_sem of the task! 177 * 178 * @returns Pointer to the mapping. 179 * (void *)-1 on failure. 180 * @param R3PtrFixed (RTR3PTR)-1 if anywhere, otherwise a specific location. 181 * @param cb The size of the mapping. 182 * @param uAlignment The alignment of the mapping. 183 * @param pTask The Linux task to create this mapping in. 184 * @param fProt The RTMEM_PROT_* mask. 185 */ 186 static void *rtR0MemObjLinuxDoMmap(RTR3PTR R3PtrFixed, size_t cb, size_t uAlignment, struct task_struct *pTask, unsigned fProt) 187 { 188 unsigned fLnxProt; 189 unsigned long ulAddr; 190 191 Assert((pTask == current)); /* do_mmap */ 192 193 /* 194 * Convert from IPRT protection to mman.h PROT_ and call do_mmap. 195 */ 196 fProt &= (RTMEM_PROT_NONE | RTMEM_PROT_READ | RTMEM_PROT_WRITE | RTMEM_PROT_EXEC); 197 if (fProt == RTMEM_PROT_NONE) 198 fLnxProt = PROT_NONE; 199 else 200 { 201 fLnxProt = 0; 202 if (fProt & RTMEM_PROT_READ) 203 fLnxProt |= PROT_READ; 204 if (fProt & RTMEM_PROT_WRITE) 205 fLnxProt |= PROT_WRITE; 206 if (fProt & RTMEM_PROT_EXEC) 207 fLnxProt |= PROT_EXEC; 208 } 209 210 if (R3PtrFixed != (RTR3PTR)-1) 211 { 212 #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 5, 0) 213 ulAddr = vm_mmap(NULL, R3PtrFixed, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS | MAP_FIXED, 0); 214 #else 215 down_write(&pTask->mm->mmap_sem); 216 ulAddr = do_mmap(NULL, R3PtrFixed, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS | MAP_FIXED, 0); 217 up_write(&pTask->mm->mmap_sem); 218 #endif 219 } 220 else 221 { 222 #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 5, 0) 223 ulAddr = vm_mmap(NULL, 0, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS, 0); 224 #else 225 down_write(&pTask->mm->mmap_sem); 226 ulAddr = do_mmap(NULL, 0, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS, 0); 227 up_write(&pTask->mm->mmap_sem); 228 #endif 229 if ( !(ulAddr & ~PAGE_MASK) 230 && (ulAddr & (uAlignment - 1))) 231 { 232 /** @todo implement uAlignment properly... We'll probably need to make some dummy mappings to fill 233 * up alignment gaps. This is of course complicated by fragmentation (which we might have cause 234 * ourselves) and further by there begin two mmap strategies (top / bottom). */ 235 /* For now, just ignore uAlignment requirements... */ 236 } 237 } 238 239 240 if (ulAddr & ~PAGE_MASK) /* ~PAGE_MASK == PAGE_OFFSET_MASK */ 241 return (void *)-1; 242 return (void *)ulAddr; 243 } 244 245 246 /** 247 * Worker that destroys a user space mapping. 248 * Undoes what rtR0MemObjLinuxDoMmap did. 249 * 250 * We acquire the mmap_sem of the task! 251 * 252 * @param pv The ring-3 mapping. 253 * @param cb The size of the mapping. 254 * @param pTask The Linux task to destroy this mapping in. 255 */ 256 static void rtR0MemObjLinuxDoMunmap(void *pv, size_t cb, struct task_struct *pTask) 257 { 258 #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 5, 0) 259 Assert(pTask == current); 260 vm_munmap((unsigned long)pv, cb); 261 #elif defined(USE_RHEL4_MUNMAP) 262 down_write(&pTask->mm->mmap_sem); 263 do_munmap(pTask->mm, (unsigned long)pv, cb, 0); /* should it be 1 or 0? */ 264 up_write(&pTask->mm->mmap_sem); 265 #else 266 down_write(&pTask->mm->mmap_sem); 267 do_munmap(pTask->mm, (unsigned long)pv, cb); 268 up_write(&pTask->mm->mmap_sem); 269 #endif 168 270 } 169 271 … … 424 526 425 527 /** 426 * Undo s what rtR0MemObjLinuxVMap() did.528 * Undoes what rtR0MemObjLinuxVMap() did. 427 529 * 428 530 * @param pMemLnx The linux memory object. … … 492 594 Assert(pTask); 493 595 if (pTask && pTask->mm) 494 { 495 down_write(&pTask->mm->mmap_sem); 496 MY_DO_MUNMAP(pTask->mm, (unsigned long)pMemLnx->Core.pv, pMemLnx->Core.cb); 497 up_write(&pTask->mm->mmap_sem); 498 } 596 rtR0MemObjLinuxDoMunmap(pMemLnx->Core.pv, pMemLnx->Core.cb, pTask); 499 597 } 500 598 else … … 517 615 Assert(pTask); 518 616 if (pTask && pTask->mm) 519 { 520 down_write(&pTask->mm->mmap_sem); 521 MY_DO_MUNMAP(pTask->mm, (unsigned long)pMemLnx->Core.pv, pMemLnx->Core.cb); 522 up_write(&pTask->mm->mmap_sem); 523 } 617 rtR0MemObjLinuxDoMunmap(pMemLnx->Core.pv, pMemLnx->Core.cb, pTask); 524 618 } 525 619 else … … 1120 1214 1121 1215 1122 /**1123 * Worker for rtR0MemObjNativeReserveUser and rtR0MemObjNativerMapUser that creates1124 * an empty user space mapping.1125 *1126 * The caller takes care of acquiring the mmap_sem of the task.1127 *1128 * @returns Pointer to the mapping.1129 * (void *)-1 on failure.1130 * @param R3PtrFixed (RTR3PTR)-1 if anywhere, otherwise a specific location.1131 * @param cb The size of the mapping.1132 * @param uAlignment The alignment of the mapping.1133 * @param pTask The Linux task to create this mapping in.1134 * @param fProt The RTMEM_PROT_* mask.1135 */1136 static void *rtR0MemObjLinuxDoMmap(RTR3PTR R3PtrFixed, size_t cb, size_t uAlignment, struct task_struct *pTask, unsigned fProt)1137 {1138 unsigned fLnxProt;1139 unsigned long ulAddr;1140 1141 /*1142 * Convert from IPRT protection to mman.h PROT_ and call do_mmap.1143 */1144 fProt &= (RTMEM_PROT_NONE | RTMEM_PROT_READ | RTMEM_PROT_WRITE | RTMEM_PROT_EXEC);1145 if (fProt == RTMEM_PROT_NONE)1146 fLnxProt = PROT_NONE;1147 else1148 {1149 fLnxProt = 0;1150 if (fProt & RTMEM_PROT_READ)1151 fLnxProt |= PROT_READ;1152 if (fProt & RTMEM_PROT_WRITE)1153 fLnxProt |= PROT_WRITE;1154 if (fProt & RTMEM_PROT_EXEC)1155 fLnxProt |= PROT_EXEC;1156 }1157 1158 if (R3PtrFixed != (RTR3PTR)-1)1159 ulAddr = do_mmap(NULL, R3PtrFixed, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS | MAP_FIXED, 0);1160 else1161 {1162 ulAddr = do_mmap(NULL, 0, cb, fLnxProt, MAP_SHARED | MAP_ANONYMOUS, 0);1163 if ( !(ulAddr & ~PAGE_MASK)1164 && (ulAddr & (uAlignment - 1)))1165 {1166 /** @todo implement uAlignment properly... We'll probably need to make some dummy mappings to fill1167 * up alignment gaps. This is of course complicated by fragmentation (which we might have cause1168 * ourselves) and further by there begin two mmap strategies (top / bottom). */1169 /* For now, just ignore uAlignment requirements... */1170 }1171 }1172 if (ulAddr & ~PAGE_MASK) /* ~PAGE_MASK == PAGE_OFFSET_MASK */1173 return (void *)-1;1174 return (void *)ulAddr;1175 }1176 1177 1178 1216 DECLHIDDEN(int) rtR0MemObjNativeReserveUser(PPRTR0MEMOBJINTERNAL ppMem, RTR3PTR R3PtrFixed, size_t cb, size_t uAlignment, RTR0PROCESS R0Process) 1179 1217 { … … 1193 1231 * Let rtR0MemObjLinuxDoMmap do the difficult bits. 1194 1232 */ 1195 down_write(&pTask->mm->mmap_sem);1196 1233 pv = rtR0MemObjLinuxDoMmap(R3PtrFixed, cb, uAlignment, pTask, RTMEM_PROT_NONE); 1197 up_write(&pTask->mm->mmap_sem);1198 1234 if (pv == (void *)-1) 1199 1235 return VERR_NO_MEMORY; … … 1202 1238 if (!pMemLnx) 1203 1239 { 1204 down_write(&pTask->mm->mmap_sem); 1205 MY_DO_MUNMAP(pTask->mm, (unsigned long)pv, cb); 1206 up_write(&pTask->mm->mmap_sem); 1240 rtR0MemObjLinuxDoMunmap(pv, cb, pTask); 1207 1241 return VERR_NO_MEMORY; 1208 1242 } … … 1391 1425 */ 1392 1426 void *pv; 1393 down_write(&pTask->mm->mmap_sem);1394 1427 pv = rtR0MemObjLinuxDoMmap(R3PtrFixed, pMemLnxToMap->Core.cb, uAlignment, pTask, fProt); 1395 1428 if (pv != (void *)-1) … … 1404 1437 size_t iPage; 1405 1438 1406 rc = 0; 1439 down_write(&pTask->mm->mmap_sem); 1440 1441 rc = VINF_SUCCESS; 1407 1442 if (pMemLnxToMap->cPages) 1408 1443 { … … 1486 1521 } 1487 1522 } 1488 if (!rc) 1489 { 1490 up_write(&pTask->mm->mmap_sem); 1523 1524 up_write(&pTask->mm->mmap_sem); 1525 1526 if (RT_SUCCESS(rc)) 1527 { 1491 1528 #ifdef VBOX_USE_PAE_HACK 1492 1529 __free_page(pDummyPage); 1493 1530 #endif 1494 1495 1531 pMemLnx->Core.pv = pv; 1496 1532 pMemLnx->Core.u.Mapping.R0Process = R0Process; … … 1502 1538 * Bail out. 1503 1539 */ 1504 MY_DO_MUNMAP(pTask->mm, (unsigned long)pv, pMemLnxToMap->Core.cb); 1505 } 1506 up_write(&pTask->mm->mmap_sem); 1540 rtR0MemObjLinuxDoMunmap(pv, pMemLnxToMap->Core.cb, pTask); 1541 } 1507 1542 rtR0MemObjDelete(&pMemLnx->Core); 1508 1543 } -
trunk/src/VBox/Runtime/r0drv/linux/the-linux-kernel.h
r39841 r41660 243 243 # if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 0) 244 244 # ifdef VM_ACCOUNT 245 # define MY_DO_MUNMAP(a,b,c) do_munmap(a, b, c, 0) /* should it be 1 or 0? */245 # define USE_RHEL4_MUNMAP 246 246 # endif 247 247 # endif … … 269 269 # endif /* !RT_ARCH_AMD64 */ 270 270 #endif /* !NO_REDHAT_HACKS */ 271 272 #ifndef MY_DO_MUNMAP273 # define MY_DO_MUNMAP(a,b,c) do_munmap(a, b, c)274 #endif275 271 276 272 #ifndef MY_CHANGE_PAGE_ATTR
Note:
See TracChangeset
for help on using the changeset viewer.