1 /* 2 * File.........: pktdrvr.c 3 * 4 * Responsible..: Gisle Vanem, giva@bgnett.no 5 * 6 * Created......: 26.Sept 1995 7 * 8 * Description..: Packet-driver interface for 16/32-bit C : 9 * Borland C/C++ 3.0+ small/large model 10 * Watcom C/C++ 11+, DOS4GW flat model 11 * Metaware HighC 3.1+ and PharLap 386|DosX 12 * GNU C/C++ 2.7+ and djgpp 2.x extender 13 * 14 * References...: PC/TCP Packet driver Specification. rev 1.09 15 * FTP Software Inc. 16 * 17 */ 18 19 #include <stdio.h> 20 #include <stdlib.h> 21 #include <string.h> 22 #include <dos.h> 23 24 #include "pcap-dos.h" 25 #include "pcap-int.h" 26 #include "msdos/pktdrvr.h" 27 28 #if (DOSX) 29 #define NUM_RX_BUF 32 /* # of buffers in Rx FIFO queue */ 30 #else 31 #define NUM_RX_BUF 10 32 #endif 33 34 #define DIM(x) (sizeof((x)) / sizeof(x[0])) 35 #define PUTS(s) do { \ 36 if (!pktInfo.quiet) \ 37 pktInfo.error ? \ 38 printf ("%s: %s\n", s, pktInfo.error) : \ 39 printf ("%s\n", pktInfo.error = s); \ 40 } while (0) 41 42 #if defined(__HIGHC__) 43 extern UINT _mwenv; 44 45 #elif defined(__DJGPP__) 46 #include <stddef.h> 47 #include <dpmi.h> 48 #include <go32.h> 49 #include <pc.h> 50 #include <sys/farptr.h> 51 52 #elif defined(__WATCOMC__) 53 #include <i86.h> 54 #include <stddef.h> 55 extern char _Extender; 56 57 #else 58 extern void far PktReceiver (void); 59 #endif 60 61 62 #if (DOSX & (DJGPP|DOS4GW)) 63 #include <sys/pack_on.h> 64 65 struct DPMI_regs { 66 DWORD r_di; 67 DWORD r_si; 68 DWORD r_bp; 69 DWORD reserved; 70 DWORD r_bx; 71 DWORD r_dx; 72 DWORD r_cx; 73 DWORD r_ax; 74 WORD r_flags; 75 WORD r_es, r_ds, r_fs, r_gs; 76 WORD r_ip, r_cs, r_sp, r_ss; 77 }; 78 79 /* Data located in a real-mode segment. This becomes far at runtime 80 */ 81 typedef struct { /* must match data/code in pkt_rx1.s */ 82 WORD _rxOutOfs; 83 WORD _rxInOfs; 84 DWORD _pktDrop; 85 BYTE _pktTemp [20]; 86 TX_ELEMENT _pktTxBuf[1]; 87 RX_ELEMENT _pktRxBuf[NUM_RX_BUF]; 88 WORD _dummy[2]; /* screenSeg,newInOffset */ 89 BYTE _fanChars[4]; 90 WORD _fanIndex; 91 BYTE _PktReceiver[15]; /* starts on a paragraph (16byte) */ 92 } PktRealStub; 93 #include <sys/pack_off.h> 94 95 static BYTE real_stub_array [] = { 96 #include "pkt_stub.inc" /* generated opcode array */ 97 }; 98 99 #define rxOutOfs offsetof (PktRealStub,_rxOutOfs) 100 #define rxInOfs offsetof (PktRealStub,_rxInOfs) 101 #define PktReceiver offsetof (PktRealStub,_PktReceiver [para_skip]) 102 #define pktDrop offsetof (PktRealStub,_pktDrop) 103 #define pktTemp offsetof (PktRealStub,_pktTemp) 104 #define pktTxBuf offsetof (PktRealStub,_pktTxBuf) 105 #define FIRST_RX_BUF offsetof (PktRealStub,_pktRxBuf [0]) 106 #define LAST_RX_BUF offsetof (PktRealStub,_pktRxBuf [NUM_RX_BUF-1]) 107 108 #else 109 extern WORD rxOutOfs; /* offsets into pktRxBuf FIFO queue */ 110 extern WORD rxInOfs; 111 extern DWORD pktDrop; /* # packets dropped in PktReceiver() */ 112 extern BYTE pktRxEnd; /* marks the end of r-mode code/data */ 113 114 extern RX_ELEMENT pktRxBuf [NUM_RX_BUF]; /* PktDrvr Rx buffers */ 115 extern TX_ELEMENT pktTxBuf; /* PktDrvr Tx buffer */ 116 extern char pktTemp[20]; /* PktDrvr temp area */ 117 118 #define FIRST_RX_BUF (WORD) &pktRxBuf [0] 119 #define LAST_RX_BUF (WORD) &pktRxBuf [NUM_RX_BUF-1] 120 #endif 121 122 123 #ifdef __BORLANDC__ /* Use Borland's inline functions */ 124 #define memcpy __memcpy__ 125 #define memcmp __memcmp__ 126 #define memset __memset__ 127 #endif 128 129 130 #if (DOSX & PHARLAP) 131 extern void PktReceiver (void); /* in pkt_rx0.asm */ 132 static int RealCopy (ULONG, ULONG, REALPTR*, FARPTR*, USHORT*); 133 134 #undef FP_SEG 135 #undef FP_OFF 136 #define FP_OFF(x) ((WORD)(x)) 137 #define FP_SEG(x) ((WORD)(realBase >> 16)) 138 #define DOS_ADDR(s,o) (((DWORD)(s) << 16) + (WORD)(o)) 139 #define r_ax eax 140 #define r_bx ebx 141 #define r_dx edx 142 #define r_cx ecx 143 #define r_si esi 144 #define r_di edi 145 #define r_ds ds 146 #define r_es es 147 LOCAL FARPTR protBase; 148 LOCAL REALPTR realBase; 149 LOCAL WORD realSeg; /* DOS para-address of allocated area */ 150 LOCAL SWI_REGS reg; 151 152 static WORD _far *rxOutOfsFp, *rxInOfsFp; 153 154 #elif (DOSX & DJGPP) 155 static _go32_dpmi_seginfo rm_mem; 156 static __dpmi_regs reg; 157 static DWORD realBase; 158 static int para_skip = 0; 159 160 #define DOS_ADDR(s,o) (((WORD)(s) << 4) + (o)) 161 #define r_ax x.ax 162 #define r_bx x.bx 163 #define r_dx x.dx 164 #define r_cx x.cx 165 #define r_si x.si 166 #define r_di x.di 167 #define r_ds x.ds 168 #define r_es x.es 169 170 #elif (DOSX & DOS4GW) 171 LOCAL struct DPMI_regs reg; 172 LOCAL WORD rm_base_seg, rm_base_sel; 173 LOCAL DWORD realBase; 174 LOCAL int para_skip = 0; 175 176 LOCAL DWORD dpmi_get_real_vector (int intr); 177 LOCAL WORD dpmi_real_malloc (int size, WORD *selector); 178 LOCAL void dpmi_real_free (WORD selector); 179 #define DOS_ADDR(s,o) (((DWORD)(s) << 4) + (WORD)(o)) 180 181 #else /* real-mode Borland etc. */ 182 static struct { 183 WORD r_ax, r_bx, r_cx, r_dx, r_bp; 184 WORD r_si, r_di, r_ds, r_es, r_flags; 185 } reg; 186 #endif 187 188 #ifdef __HIGHC__ 189 #pragma Alias (pktDrop, "_pktDrop") 190 #pragma Alias (pktRxBuf, "_pktRxBuf") 191 #pragma Alias (pktTxBuf, "_pktTxBuf") 192 #pragma Alias (pktTemp, "_pktTemp") 193 #pragma Alias (rxOutOfs, "_rxOutOfs") 194 #pragma Alias (rxInOfs, "_rxInOfs") 195 #pragma Alias (pktRxEnd, "_pktRxEnd") 196 #pragma Alias (PktReceiver,"_PktReceiver") 197 #endif 198 199 200 PUBLIC PKT_STAT pktStat; /* statistics for packets */ 201 PUBLIC PKT_INFO pktInfo; /* packet-driver information */ 202 203 PUBLIC PKT_RX_MODE receiveMode = PDRX_DIRECT; 204 PUBLIC ETHER myAddress = { 0, 0, 0, 0, 0, 0 }; 205 PUBLIC ETHER ethBroadcast = { 255,255,255,255,255,255 }; 206 207 LOCAL struct { /* internal statistics */ 208 DWORD tooSmall; /* size < ETH_MIN */ 209 DWORD tooLarge; /* size > ETH_MAX */ 210 DWORD badSync; /* count_1 != count_2 */ 211 DWORD wrongHandle; /* upcall to wrong handle */ 212 } intStat; 213 214 /***************************************************************************/ 215 216 PUBLIC const char *PktGetErrorStr (int errNum) 217 { 218 static const char *errStr[] = { 219 "", 220 "Invalid handle number", 221 "No interfaces of specified class found", 222 "No interfaces of specified type found", 223 "No interfaces of specified number found", 224 "Bad packet type specified", 225 "Interface does not support multicast", 226 "Packet driver cannot terminate", 227 "Invalid receiver mode specified", 228 "Insufficient memory space", 229 "Type previously accessed, and not released", 230 "Command out of range, or not implemented", 231 "Cannot send packet (usually hardware error)", 232 "Cannot change hardware address ( > 1 handle open)", 233 "Hardware address has bad length or format", 234 "Cannot reset interface (more than 1 handle open)", 235 "Bad Check-sum", 236 "Bad size", 237 "Bad sync" , 238 "Source hit" 239 }; 240 241 if (errNum < 0 || errNum >= DIM(errStr)) 242 return ("Unknown driver error."); 243 return (errStr [errNum]); 244 } 245 246 /**************************************************************************/ 247 248 PUBLIC const char *PktGetClassName (WORD class) 249 { 250 switch (class) 251 { 252 case PD_ETHER: 253 return ("DIX-Ether"); 254 case PD_PRONET10: 255 return ("ProNET-10"); 256 case PD_IEEE8025: 257 return ("IEEE 802.5"); 258 case PD_OMNINET: 259 return ("OmniNet"); 260 case PD_APPLETALK: 261 return ("AppleTalk"); 262 case PD_SLIP: 263 return ("SLIP"); 264 case PD_STARTLAN: 265 return ("StartLAN"); 266 case PD_ARCNET: 267 return ("ArcNet"); 268 case PD_AX25: 269 return ("AX.25"); 270 case PD_KISS: 271 return ("KISS"); 272 case PD_IEEE8023_2: 273 return ("IEEE 802.3 w/802.2 hdr"); 274 case PD_FDDI8022: 275 return ("FDDI w/802.2 hdr"); 276 case PD_X25: 277 return ("X.25"); 278 case PD_LANstar: 279 return ("LANstar"); 280 case PD_PPP: 281 return ("PPP"); 282 default: 283 return ("unknown"); 284 } 285 } 286 287 /**************************************************************************/ 288 289 PUBLIC char const *PktRXmodeStr (PKT_RX_MODE mode) 290 { 291 static const char *modeStr [] = { 292 "Receiver turned off", 293 "Receive only directly addressed packets", 294 "Receive direct & broadcast packets", 295 "Receive direct,broadcast and limited multicast packets", 296 "Receive direct,broadcast and all multicast packets", 297 "Receive all packets (promiscuous mode)" 298 }; 299 300 if (mode > DIM(modeStr)) 301 return ("??"); 302 return (modeStr [mode-1]); 303 } 304 305 /**************************************************************************/ 306 307 LOCAL __inline BOOL PktInterrupt (void) 308 { 309 BOOL okay; 310 311 #if (DOSX & PHARLAP) 312 _dx_real_int ((UINT)pktInfo.intr, ®); 313 okay = ((reg.flags & 1) == 0); /* OK if carry clear */ 314 315 #elif (DOSX & DJGPP) 316 __dpmi_int ((int)pktInfo.intr, ®); 317 okay = ((reg.x.flags & 1) == 0); 318 319 #elif (DOSX & DOS4GW) 320 union REGS r; 321 struct SREGS s; 322 323 memset (&r, 0, sizeof(r)); 324 segread (&s); 325 r.w.ax = 0x300; 326 r.x.ebx = pktInfo.intr; 327 r.w.cx = 0; 328 s.es = FP_SEG (®); 329 r.x.edi = FP_OFF (®); 330 reg.r_flags = 0; 331 reg.r_ss = reg.r_sp = 0; /* DPMI host provides stack */ 332 333 int386x (0x31, &r, &r, &s); 334 okay = (!r.w.cflag); 335 336 #else 337 reg.r_flags = 0; 338 intr (pktInfo.intr, (struct REGPACK*)®); 339 okay = ((reg.r_flags & 1) == 0); 340 #endif 341 342 if (okay) 343 pktInfo.error = NULL; 344 else pktInfo.error = PktGetErrorStr (reg.r_dx >> 8); 345 return (okay); 346 } 347 348 /**************************************************************************/ 349 350 /* 351 * Search for packet driver at interrupt 60h through 80h. If ASCIIZ 352 * string "PKT DRVR" found at offset 3 in the interrupt handler, return 353 * interrupt number, else return zero in pktInfo.intr 354 */ 355 PUBLIC BOOL PktSearchDriver (void) 356 { 357 BYTE intr = 0x20; 358 BOOL found = FALSE; 359 360 while (!found && intr < 0xFF) 361 { 362 static char str[12]; /* 3 + strlen("PKT DRVR") */ 363 static char pktStr[9] = "PKT DRVR"; /* ASCIIZ string at ofs 3 */ 364 DWORD rp; /* in interrupt routine */ 365 366 #if (DOSX & PHARLAP) 367 _dx_rmiv_get (intr, &rp); 368 ReadRealMem (&str, (REALPTR)rp, sizeof(str)); 369 370 #elif (DOSX & DJGPP) 371 __dpmi_raddr realAdr; 372 __dpmi_get_real_mode_interrupt_vector (intr, &realAdr); 373 rp = (realAdr.segment << 4) + realAdr.offset16; 374 dosmemget (rp, sizeof(str), &str); 375 376 #elif (DOSX & DOS4GW) 377 rp = dpmi_get_real_vector (intr); 378 memcpy (&str, (void*)rp, sizeof(str)); 379 380 #else 381 _fmemcpy (&str, getvect(intr), sizeof(str)); 382 #endif 383 384 found = memcmp (&str[3],&pktStr,sizeof(pktStr)) == 0; 385 intr++; 386 } 387 pktInfo.intr = (found ? intr-1 : 0); 388 return (found); 389 } 390 391 392 /**************************************************************************/ 393 394 static BOOL PktSetAccess (void) 395 { 396 reg.r_ax = 0x0200 + pktInfo.class; 397 reg.r_bx = 0xFFFF; 398 reg.r_dx = 0; 399 reg.r_cx = 0; 400 401 #if (DOSX & PHARLAP) 402 reg.ds = 0; 403 reg.esi = 0; 404 reg.es = RP_SEG (realBase); 405 reg.edi = (WORD) &PktReceiver; 406 407 #elif (DOSX & DJGPP) 408 reg.x.ds = 0; 409 reg.x.si = 0; 410 reg.x.es = rm_mem.rm_segment; 411 reg.x.di = PktReceiver; 412 413 #elif (DOSX & DOS4GW) 414 reg.r_ds = 0; 415 reg.r_si = 0; 416 reg.r_es = rm_base_seg; 417 reg.r_di = PktReceiver; 418 419 #else 420 reg.r_ds = 0; 421 reg.r_si = 0; 422 reg.r_es = FP_SEG (&PktReceiver); 423 reg.r_di = FP_OFF (&PktReceiver); 424 #endif 425 426 if (!PktInterrupt()) 427 return (FALSE); 428 429 pktInfo.handle = reg.r_ax; 430 return (TRUE); 431 } 432 433 /**************************************************************************/ 434 435 PUBLIC BOOL PktReleaseHandle (WORD handle) 436 { 437 reg.r_ax = 0x0300; 438 reg.r_bx = handle; 439 return PktInterrupt(); 440 } 441 442 /**************************************************************************/ 443 444 PUBLIC BOOL PktTransmit (const void *eth, int len) 445 { 446 if (len > ETH_MTU) 447 return (FALSE); 448 449 reg.r_ax = 0x0400; /* Function 4, send pkt */ 450 reg.r_cx = len; /* total size of frame */ 451 452 #if (DOSX & DJGPP) 453 dosmemput (eth, len, realBase+pktTxBuf); 454 reg.x.ds = rm_mem.rm_segment; /* DOS data segment and */ 455 reg.x.si = pktTxBuf; /* DOS offset to buffer */ 456 457 #elif (DOSX & DOS4GW) 458 memcpy ((void*)(realBase+pktTxBuf), eth, len); 459 reg.r_ds = rm_base_seg; 460 reg.r_si = pktTxBuf; 461 462 #elif (DOSX & PHARLAP) 463 memcpy (&pktTxBuf, eth, len); 464 reg.r_ds = FP_SEG (&pktTxBuf); 465 reg.r_si = FP_OFF (&pktTxBuf); 466 467 #else 468 reg.r_ds = FP_SEG (eth); 469 reg.r_si = FP_OFF (eth); 470 #endif 471 472 return PktInterrupt(); 473 } 474 475 /**************************************************************************/ 476 477 #if (DOSX & (DJGPP|DOS4GW)) 478 LOCAL __inline BOOL CheckElement (RX_ELEMENT *rx) 479 #else 480 LOCAL __inline BOOL CheckElement (RX_ELEMENT _far *rx) 481 #endif 482 { 483 WORD count_1, count_2; 484 485 /* 486 * We got an upcall to the same RMCB with wrong handle. 487 * This can happen if we failed to release handle at program exit 488 */ 489 if (rx->handle != pktInfo.handle) 490 { 491 pktInfo.error = "Wrong handle"; 492 intStat.wrongHandle++; 493 PktReleaseHandle (rx->handle); 494 return (FALSE); 495 } 496 count_1 = rx->firstCount; 497 count_2 = rx->secondCount; 498 499 if (count_1 != count_2) 500 { 501 pktInfo.error = "Bad sync"; 502 intStat.badSync++; 503 return (FALSE); 504 } 505 if (count_1 > ETH_MAX) 506 { 507 pktInfo.error = "Large esize"; 508 intStat.tooLarge++; 509 return (FALSE); 510 } 511 #if 0 512 if (count_1 < ETH_MIN) 513 { 514 pktInfo.error = "Small esize"; 515 intStat.tooSmall++; 516 return (FALSE); 517 } 518 #endif 519 return (TRUE); 520 } 521 522 /**************************************************************************/ 523 524 PUBLIC BOOL PktTerminHandle (WORD handle) 525 { 526 reg.r_ax = 0x0500; 527 reg.r_bx = handle; 528 return PktInterrupt(); 529 } 530 531 /**************************************************************************/ 532 533 PUBLIC BOOL PktResetInterface (WORD handle) 534 { 535 reg.r_ax = 0x0700; 536 reg.r_bx = handle; 537 return PktInterrupt(); 538 } 539 540 /**************************************************************************/ 541 542 PUBLIC BOOL PktSetReceiverMode (PKT_RX_MODE mode) 543 { 544 if (pktInfo.class == PD_SLIP || pktInfo.class == PD_PPP) 545 return (TRUE); 546 547 reg.r_ax = 0x1400; 548 reg.r_bx = pktInfo.handle; 549 reg.r_cx = (WORD)mode; 550 551 if (!PktInterrupt()) 552 return (FALSE); 553 554 receiveMode = mode; 555 return (TRUE); 556 } 557 558 /**************************************************************************/ 559 560 PUBLIC BOOL PktGetReceiverMode (PKT_RX_MODE *mode) 561 { 562 reg.r_ax = 0x1500; 563 reg.r_bx = pktInfo.handle; 564 565 if (!PktInterrupt()) 566 return (FALSE); 567 568 *mode = reg.r_ax; 569 return (TRUE); 570 } 571 572 /**************************************************************************/ 573 574 static PKT_STAT initialStat; /* statistics at startup */ 575 static BOOL resetStat = FALSE; /* statistics reset ? */ 576 577 PUBLIC BOOL PktGetStatistics (WORD handle) 578 { 579 reg.r_ax = 0x1800; 580 reg.r_bx = handle; 581 582 if (!PktInterrupt()) 583 return (FALSE); 584 585 #if (DOSX & PHARLAP) 586 ReadRealMem (&pktStat, DOS_ADDR(reg.ds,reg.esi), sizeof(pktStat)); 587 588 #elif (DOSX & DJGPP) 589 dosmemget (DOS_ADDR(reg.x.ds,reg.x.si), sizeof(pktStat), &pktStat); 590 591 #elif (DOSX & DOS4GW) 592 memcpy (&pktStat, (void*)DOS_ADDR(reg.r_ds,reg.r_si), sizeof(pktStat)); 593 594 #else 595 _fmemcpy (&pktStat, MK_FP(reg.r_ds,reg.r_si), sizeof(pktStat)); 596 #endif 597 598 return (TRUE); 599 } 600 601 /**************************************************************************/ 602 603 PUBLIC BOOL PktSessStatistics (WORD handle) 604 { 605 if (!PktGetStatistics(pktInfo.handle)) 606 return (FALSE); 607 608 if (resetStat) 609 { 610 pktStat.inPackets -= initialStat.inPackets; 611 pktStat.outPackets -= initialStat.outPackets; 612 pktStat.inBytes -= initialStat.inBytes; 613 pktStat.outBytes -= initialStat.outBytes; 614 pktStat.inErrors -= initialStat.inErrors; 615 pktStat.outErrors -= initialStat.outErrors; 616 pktStat.outErrors -= initialStat.outErrors; 617 pktStat.lost -= initialStat.lost; 618 } 619 return (TRUE); 620 } 621 622 /**************************************************************************/ 623 624 PUBLIC BOOL PktResetStatistics (WORD handle) 625 { 626 if (!PktGetStatistics(pktInfo.handle)) 627 return (FALSE); 628 629 memcpy (&initialStat, &pktStat, sizeof(initialStat)); 630 resetStat = TRUE; 631 return (TRUE); 632 } 633 634 /**************************************************************************/ 635 636 PUBLIC BOOL PktGetAddress (ETHER *addr) 637 { 638 reg.r_ax = 0x0600; 639 reg.r_bx = pktInfo.handle; 640 reg.r_cx = sizeof (*addr); 641 642 #if (DOSX & DJGPP) 643 reg.x.es = rm_mem.rm_segment; 644 reg.x.di = pktTemp; 645 #elif (DOSX & DOS4GW) 646 reg.r_es = rm_base_seg; 647 reg.r_di = pktTemp; 648 #else 649 reg.r_es = FP_SEG (&pktTemp); 650 reg.r_di = FP_OFF (&pktTemp); /* ES:DI = address for result */ 651 #endif 652 653 if (!PktInterrupt()) 654 return (FALSE); 655 656 #if (DOSX & PHARLAP) 657 ReadRealMem (addr, realBase + (WORD)&pktTemp, sizeof(*addr)); 658 659 #elif (DOSX & DJGPP) 660 dosmemget (realBase+pktTemp, sizeof(*addr), addr); 661 662 #elif (DOSX & DOS4GW) 663 memcpy (addr, (void*)(realBase+pktTemp), sizeof(*addr)); 664 665 #else 666 memcpy ((void*)addr, &pktTemp, sizeof(*addr)); 667 #endif 668 669 return (TRUE); 670 } 671 672 /**************************************************************************/ 673 674 PUBLIC BOOL PktSetAddress (const ETHER *addr) 675 { 676 /* copy addr to real-mode scrath area */ 677 678 #if (DOSX & PHARLAP) 679 WriteRealMem (realBase + (WORD)&pktTemp, (void*)addr, sizeof(*addr)); 680 681 #elif (DOSX & DJGPP) 682 dosmemput (addr, sizeof(*addr), realBase+pktTemp); 683 684 #elif (DOSX & DOS4GW) 685 memcpy ((void*)(realBase+pktTemp), addr, sizeof(*addr)); 686 687 #else 688 memcpy (&pktTemp, (void*)addr, sizeof(*addr)); 689 #endif 690 691 reg.r_ax = 0x1900; 692 reg.r_cx = sizeof (*addr); /* address length */ 693 694 #if (DOSX & DJGPP) 695 reg.x.es = rm_mem.rm_segment; /* DOS offset to param */ 696 reg.x.di = pktTemp; /* DOS segment to param */ 697 #elif (DOSX & DOS4GW) 698 reg.r_es = rm_base_seg; 699 reg.r_di = pktTemp; 700 #else 701 reg.r_es = FP_SEG (&pktTemp); 702 reg.r_di = FP_OFF (&pktTemp); 703 #endif 704 705 return PktInterrupt(); 706 } 707 708 /**************************************************************************/ 709 710 PUBLIC BOOL PktGetDriverInfo (void) 711 { 712 pktInfo.majVer = 0; 713 pktInfo.minVer = 0; 714 memset (&pktInfo.name, 0, sizeof(pktInfo.name)); 715 reg.r_ax = 0x01FF; 716 reg.r_bx = 0; 717 718 if (!PktInterrupt()) 719 return (FALSE); 720 721 pktInfo.number = reg.r_cx & 0xFF; 722 pktInfo.class = reg.r_cx >> 8; 723 #if 0 724 pktInfo.minVer = reg.r_bx % 10; 725 pktInfo.majVer = reg.r_bx / 10; 726 #else 727 pktInfo.majVer = reg.r_bx; // !! 728 #endif 729 pktInfo.funcs = reg.r_ax & 0xFF; 730 pktInfo.type = reg.r_dx & 0xFF; 731 732 #if (DOSX & PHARLAP) 733 ReadRealMem (&pktInfo.name, DOS_ADDR(reg.ds,reg.esi), sizeof(pktInfo.name)); 734 735 #elif (DOSX & DJGPP) 736 dosmemget (DOS_ADDR(reg.x.ds,reg.x.si), sizeof(pktInfo.name), &pktInfo.name); 737 738 #elif (DOSX & DOS4GW) 739 memcpy (&pktInfo.name, (void*)DOS_ADDR(reg.r_ds,reg.r_si), sizeof(pktInfo.name)); 740 741 #else 742 _fmemcpy (&pktInfo.name, MK_FP(reg.r_ds,reg.r_si), sizeof(pktInfo.name)); 743 #endif 744 return (TRUE); 745 } 746 747 /**************************************************************************/ 748 749 PUBLIC BOOL PktGetDriverParam (void) 750 { 751 reg.r_ax = 0x0A00; 752 753 if (!PktInterrupt()) 754 return (FALSE); 755 756 #if (DOSX & PHARLAP) 757 ReadRealMem (&pktInfo.majVer, DOS_ADDR(reg.es,reg.edi), PKT_PARAM_SIZE); 758 759 #elif (DOSX & DJGPP) 760 dosmemget (DOS_ADDR(reg.x.es,reg.x.di), PKT_PARAM_SIZE, &pktInfo.majVer); 761 762 #elif (DOSX & DOS4GW) 763 memcpy (&pktInfo.majVer, (void*)DOS_ADDR(reg.r_es,reg.r_di), PKT_PARAM_SIZE); 764 765 #else 766 _fmemcpy (&pktInfo.majVer, MK_FP(reg.r_es,reg.r_di), PKT_PARAM_SIZE); 767 #endif 768 return (TRUE); 769 } 770 771 /**************************************************************************/ 772 773 #if (DOSX & PHARLAP) 774 PUBLIC int PktReceive (BYTE *buf, int max) 775 { 776 WORD inOfs = *rxInOfsFp; 777 WORD outOfs = *rxOutOfsFp; 778 779 if (outOfs != inOfs) 780 { 781 RX_ELEMENT _far *head = (RX_ELEMENT _far*)(protBase+outOfs); 782 int size, len = max; 783 784 if (CheckElement(head)) 785 { 786 size = min (head->firstCount, sizeof(RX_ELEMENT)); 787 len = min (size, max); 788 _fmemcpy (buf, &head->destin, len); 789 } 790 else 791 size = -1; 792 793 outOfs += sizeof (RX_ELEMENT); 794 if (outOfs > LAST_RX_BUF) 795 outOfs = FIRST_RX_BUF; 796 *rxOutOfsFp = outOfs; 797 return (size); 798 } 799 return (0); 800 } 801 802 PUBLIC void PktQueueBusy (BOOL busy) 803 { 804 *rxOutOfsFp = busy ? (*rxInOfsFp + sizeof(RX_ELEMENT)) : *rxInOfsFp; 805 if (*rxOutOfsFp > LAST_RX_BUF) 806 *rxOutOfsFp = FIRST_RX_BUF; 807 *(DWORD _far*)(protBase + (WORD)&pktDrop) = 0; 808 } 809 810 PUBLIC WORD PktBuffersUsed (void) 811 { 812 WORD inOfs = *rxInOfsFp; 813 WORD outOfs = *rxOutOfsFp; 814 815 if (inOfs >= outOfs) 816 return (inOfs - outOfs) / sizeof(RX_ELEMENT); 817 return (NUM_RX_BUF - (outOfs - inOfs) / sizeof(RX_ELEMENT)); 818 } 819 820 PUBLIC DWORD PktRxDropped (void) 821 { 822 return (*(DWORD _far*)(protBase + (WORD)&pktDrop)); 823 } 824 825 #elif (DOSX & DJGPP) 826 PUBLIC int PktReceive (BYTE *buf, int max) 827 { 828 WORD ofs = _farpeekw (_dos_ds, realBase+rxOutOfs); 829 830 if (ofs != _farpeekw (_dos_ds, realBase+rxInOfs)) 831 { 832 RX_ELEMENT head; 833 int size, len = max; 834 835 head.firstCount = _farpeekw (_dos_ds, realBase+ofs); 836 head.secondCount = _farpeekw (_dos_ds, realBase+ofs+2); 837 head.handle = _farpeekw (_dos_ds, realBase+ofs+4); 838 839 if (CheckElement(&head)) 840 { 841 size = min (head.firstCount, sizeof(RX_ELEMENT)); 842 len = min (size, max); 843 dosmemget (realBase+ofs+6, len, buf); 844 } 845 else 846 size = -1; 847 848 ofs += sizeof (RX_ELEMENT); 849 if (ofs > LAST_RX_BUF) 850 _farpokew (_dos_ds, realBase+rxOutOfs, FIRST_RX_BUF); 851 else _farpokew (_dos_ds, realBase+rxOutOfs, ofs); 852 return (size); 853 } 854 return (0); 855 } 856 857 PUBLIC void PktQueueBusy (BOOL busy) 858 { 859 WORD ofs; 860 861 disable(); 862 ofs = _farpeekw (_dos_ds, realBase+rxInOfs); 863 if (busy) 864 ofs += sizeof (RX_ELEMENT); 865 866 if (ofs > LAST_RX_BUF) 867 _farpokew (_dos_ds, realBase+rxOutOfs, FIRST_RX_BUF); 868 else _farpokew (_dos_ds, realBase+rxOutOfs, ofs); 869 _farpokel (_dos_ds, realBase+pktDrop, 0UL); 870 enable(); 871 } 872 873 PUBLIC WORD PktBuffersUsed (void) 874 { 875 WORD inOfs, outOfs; 876 877 disable(); 878 inOfs = _farpeekw (_dos_ds, realBase+rxInOfs); 879 outOfs = _farpeekw (_dos_ds, realBase+rxOutOfs); 880 enable(); 881 if (inOfs >= outOfs) 882 return (inOfs - outOfs) / sizeof(RX_ELEMENT); 883 return (NUM_RX_BUF - (outOfs - inOfs) / sizeof(RX_ELEMENT)); 884 } 885 886 PUBLIC DWORD PktRxDropped (void) 887 { 888 return _farpeekl (_dos_ds, realBase+pktDrop); 889 } 890 891 #elif (DOSX & DOS4GW) 892 PUBLIC int PktReceive (BYTE *buf, int max) 893 { 894 WORD ofs = *(WORD*) (realBase+rxOutOfs); 895 896 if (ofs != *(WORD*) (realBase+rxInOfs)) 897 { 898 RX_ELEMENT head; 899 int size, len = max; 900 901 head.firstCount = *(WORD*) (realBase+ofs); 902 head.secondCount = *(WORD*) (realBase+ofs+2); 903 head.handle = *(WORD*) (realBase+ofs+4); 904 905 if (CheckElement(&head)) 906 { 907 size = min (head.firstCount, sizeof(RX_ELEMENT)); 908 len = min (size, max); 909 memcpy (buf, (const void*)(realBase+ofs+6), len); 910 } 911 else 912 size = -1; 913 914 ofs += sizeof (RX_ELEMENT); 915 if (ofs > LAST_RX_BUF) 916 *(WORD*) (realBase+rxOutOfs) = FIRST_RX_BUF; 917 else *(WORD*) (realBase+rxOutOfs) = ofs; 918 return (size); 919 } 920 return (0); 921 } 922 923 PUBLIC void PktQueueBusy (BOOL busy) 924 { 925 WORD ofs; 926 927 _disable(); 928 ofs = *(WORD*) (realBase+rxInOfs); 929 if (busy) 930 ofs += sizeof (RX_ELEMENT); 931 932 if (ofs > LAST_RX_BUF) 933 *(WORD*) (realBase+rxOutOfs) = FIRST_RX_BUF; 934 else *(WORD*) (realBase+rxOutOfs) = ofs; 935 *(DWORD*) (realBase+pktDrop) = 0UL; 936 _enable(); 937 } 938 939 PUBLIC WORD PktBuffersUsed (void) 940 { 941 WORD inOfs, outOfs; 942 943 _disable(); 944 inOfs = *(WORD*) (realBase+rxInOfs); 945 outOfs = *(WORD*) (realBase+rxOutOfs); 946 _enable(); 947 if (inOfs >= outOfs) 948 return (inOfs - outOfs) / sizeof(RX_ELEMENT); 949 return (NUM_RX_BUF - (outOfs - inOfs) / sizeof(RX_ELEMENT)); 950 } 951 952 PUBLIC DWORD PktRxDropped (void) 953 { 954 return *(DWORD*) (realBase+pktDrop); 955 } 956 957 #else /* real-mode small/large model */ 958 959 PUBLIC int PktReceive (BYTE *buf, int max) 960 { 961 if (rxOutOfs != rxInOfs) 962 { 963 RX_ELEMENT far *head = (RX_ELEMENT far*) MK_FP (_DS,rxOutOfs); 964 int size, len = max; 965 966 if (CheckElement(head)) 967 { 968 size = min (head->firstCount, sizeof(RX_ELEMENT)); 969 len = min (size, max); 970 _fmemcpy (buf, &head->destin, len); 971 } 972 else 973 size = -1; 974 975 rxOutOfs += sizeof (RX_ELEMENT); 976 if (rxOutOfs > LAST_RX_BUF) 977 rxOutOfs = FIRST_RX_BUF; 978 return (size); 979 } 980 return (0); 981 } 982 983 PUBLIC void PktQueueBusy (BOOL busy) 984 { 985 rxOutOfs = busy ? (rxInOfs + sizeof(RX_ELEMENT)) : rxInOfs; 986 if (rxOutOfs > LAST_RX_BUF) 987 rxOutOfs = FIRST_RX_BUF; 988 pktDrop = 0L; 989 } 990 991 PUBLIC WORD PktBuffersUsed (void) 992 { 993 WORD inOfs = rxInOfs; 994 WORD outOfs = rxOutOfs; 995 996 if (inOfs >= outOfs) 997 return ((inOfs - outOfs) / sizeof(RX_ELEMENT)); 998 return (NUM_RX_BUF - (outOfs - inOfs) / sizeof(RX_ELEMENT)); 999 } 1000 1001 PUBLIC DWORD PktRxDropped (void) 1002 { 1003 return (pktDrop); 1004 } 1005 #endif 1006 1007 /**************************************************************************/ 1008 1009 LOCAL __inline void PktFreeMem (void) 1010 { 1011 #if (DOSX & PHARLAP) 1012 if (realSeg) 1013 { 1014 _dx_real_free (realSeg); 1015 realSeg = 0; 1016 } 1017 #elif (DOSX & DJGPP) 1018 if (rm_mem.rm_segment) 1019 { 1020 unsigned ofs; /* clear the DOS-mem to prevent further upcalls */ 1021 1022 for (ofs = 0; ofs < 16 * rm_mem.size / 4; ofs += 4) 1023 _farpokel (_dos_ds, realBase + ofs, 0); 1024 _go32_dpmi_free_dos_memory (&rm_mem); 1025 rm_mem.rm_segment = 0; 1026 } 1027 #elif (DOSX & DOS4GW) 1028 if (rm_base_sel) 1029 { 1030 dpmi_real_free (rm_base_sel); 1031 rm_base_sel = 0; 1032 } 1033 #endif 1034 } 1035 1036 /**************************************************************************/ 1037 1038 PUBLIC BOOL PktExitDriver (void) 1039 { 1040 if (pktInfo.handle) 1041 { 1042 if (!PktSetReceiverMode(PDRX_BROADCAST)) 1043 PUTS ("Error restoring receiver mode."); 1044 1045 if (!PktReleaseHandle(pktInfo.handle)) 1046 PUTS ("Error releasing PKT-DRVR handle."); 1047 1048 PktFreeMem(); 1049 pktInfo.handle = 0; 1050 } 1051 1052 if (pcap_pkt_debug >= 1) 1053 printf ("Internal stats: too-small %lu, too-large %lu, bad-sync %lu, " 1054 "wrong-handle %lu\n", 1055 intStat.tooSmall, intStat.tooLarge, 1056 intStat.badSync, intStat.wrongHandle); 1057 return (TRUE); 1058 } 1059 1060 #if (DOSX & (DJGPP|DOS4GW)) 1061 static void dump_pkt_stub (void) 1062 { 1063 int i; 1064 1065 fprintf (stderr, "PktReceiver %lu, pkt_stub[PktReceiver] =\n", 1066 PktReceiver); 1067 for (i = 0; i < 15; i++) 1068 fprintf (stderr, "%02X, ", real_stub_array[i+PktReceiver]); 1069 fputs ("\n", stderr); 1070 } 1071 #endif 1072 1073 /* 1074 * Front end initialization routine 1075 */ 1076 PUBLIC BOOL PktInitDriver (PKT_RX_MODE mode) 1077 { 1078 PKT_RX_MODE rxMode; 1079 BOOL writeInfo = (pcap_pkt_debug >= 3); 1080 1081 pktInfo.quiet = (pcap_pkt_debug < 3); 1082 1083 #if (DOSX & PHARLAP) && defined(__HIGHC__) 1084 if (_mwenv != 2) 1085 { 1086 fprintf (stderr, "Only Pharlap DOS extender supported.\n"); 1087 return (FALSE); 1088 } 1089 #endif 1090 1091 #if (DOSX & PHARLAP) && defined(__WATCOMC__) 1092 if (_Extender != 1) 1093 { 1094 fprintf (stderr, "Only DOS4GW style extenders supported.\n"); 1095 return (FALSE); 1096 } 1097 #endif 1098 1099 if (!PktSearchDriver()) 1100 { 1101 PUTS ("Packet driver not found."); 1102 PktFreeMem(); 1103 return (FALSE); 1104 } 1105 1106 if (!PktGetDriverInfo()) 1107 { 1108 PUTS ("Error getting pkt-drvr information."); 1109 PktFreeMem(); 1110 return (FALSE); 1111 } 1112 1113 #if (DOSX & PHARLAP) 1114 if (RealCopy((ULONG)&rxOutOfs, (ULONG)&pktRxEnd, 1115 &realBase, &protBase, (USHORT*)&realSeg)) 1116 { 1117 rxOutOfsFp = (WORD _far *) (protBase + (WORD) &rxOutOfs); 1118 rxInOfsFp = (WORD _far *) (protBase + (WORD) &rxInOfs); 1119 *rxOutOfsFp = FIRST_RX_BUF; 1120 *rxInOfsFp = FIRST_RX_BUF; 1121 } 1122 else 1123 { 1124 PUTS ("Cannot allocate real-mode stub."); 1125 return (FALSE); 1126 } 1127 1128 #elif (DOSX & (DJGPP|DOS4GW)) 1129 if (sizeof(real_stub_array) > 0xFFFF) 1130 { 1131 fprintf (stderr, "`real_stub_array[]' too big.\n"); 1132 return (FALSE); 1133 } 1134 #if (DOSX & DJGPP) 1135 rm_mem.size = (sizeof(real_stub_array) + 15) / 16; 1136 1137 if (_go32_dpmi_allocate_dos_memory(&rm_mem) || rm_mem.rm_offset != 0) 1138 { 1139 PUTS ("real-mode init failed."); 1140 return (FALSE); 1141 } 1142 realBase = (rm_mem.rm_segment << 4); 1143 dosmemput (&real_stub_array, sizeof(real_stub_array), realBase); 1144 _farpokel (_dos_ds, realBase+rxOutOfs, FIRST_RX_BUF); 1145 _farpokel (_dos_ds, realBase+rxInOfs, FIRST_RX_BUF); 1146 1147 #elif (DOSX & DOS4GW) 1148 rm_base_seg = dpmi_real_malloc (sizeof(real_stub_array), &rm_base_sel); 1149 if (!rm_base_seg) 1150 { 1151 PUTS ("real-mode init failed."); 1152 return (FALSE); 1153 } 1154 realBase = (rm_base_seg << 4); 1155 memcpy ((void*)realBase, &real_stub_array, sizeof(real_stub_array)); 1156 *(WORD*) (realBase+rxOutOfs) = FIRST_RX_BUF; 1157 *(WORD*) (realBase+rxInOfs) = FIRST_RX_BUF; 1158 1159 #endif 1160 { 1161 int pushf = PktReceiver; 1162 1163 while (real_stub_array[pushf++] != 0x9C && /* pushf */ 1164 real_stub_array[pushf] != 0xFA) /* cli */ 1165 { 1166 if (++para_skip > 16) 1167 { 1168 fprintf (stderr, "Something wrong with `pkt_stub.inc'.\n"); 1169 para_skip = 0; 1170 dump_pkt_stub(); 1171 return (FALSE); 1172 } 1173 } 1174 if (*(WORD*)(real_stub_array + offsetof(PktRealStub,_dummy)) != 0xB800) 1175 { 1176 fprintf (stderr, "`real_stub_array[]' is misaligned.\n"); 1177 return (FALSE); 1178 } 1179 } 1180 1181 if (pcap_pkt_debug > 2) 1182 dump_pkt_stub(); 1183 1184 #else 1185 rxOutOfs = FIRST_RX_BUF; 1186 rxInOfs = FIRST_RX_BUF; 1187 #endif 1188 1189 if (!PktSetAccess()) 1190 { 1191 PUTS ("Error setting pkt-drvr access."); 1192 PktFreeMem(); 1193 return (FALSE); 1194 } 1195 1196 if (!PktGetAddress(&myAddress)) 1197 { 1198 PUTS ("Error fetching adapter address."); 1199 PktFreeMem(); 1200 return (FALSE); 1201 } 1202 1203 if (!PktSetReceiverMode(mode)) 1204 { 1205 PUTS ("Error setting receiver mode."); 1206 PktFreeMem(); 1207 return (FALSE); 1208 } 1209 1210 if (!PktGetReceiverMode(&rxMode)) 1211 { 1212 PUTS ("Error getting receiver mode."); 1213 PktFreeMem(); 1214 return (FALSE); 1215 } 1216 1217 if (writeInfo) 1218 printf ("Pkt-driver information:\n" 1219 " Version : %d.%d\n" 1220 " Name : %.15s\n" 1221 " Class : %u (%s)\n" 1222 " Type : %u\n" 1223 " Number : %u\n" 1224 " Funcs : %u\n" 1225 " Intr : %Xh\n" 1226 " Handle : %u\n" 1227 " Extended : %s\n" 1228 " Hi-perf : %s\n" 1229 " RX mode : %s\n" 1230 " Eth-addr : %02X:%02X:%02X:%02X:%02X:%02X\n", 1231 1232 pktInfo.majVer, pktInfo.minVer, pktInfo.name, 1233 pktInfo.class, PktGetClassName(pktInfo.class), 1234 pktInfo.type, pktInfo.number, 1235 pktInfo.funcs, pktInfo.intr, pktInfo.handle, 1236 pktInfo.funcs == 2 || pktInfo.funcs == 6 ? "Yes" : "No", 1237 pktInfo.funcs == 5 || pktInfo.funcs == 6 ? "Yes" : "No", 1238 PktRXmodeStr(rxMode), 1239 myAddress[0], myAddress[1], myAddress[2], 1240 myAddress[3], myAddress[4], myAddress[5]); 1241 1242 #if defined(DEBUG) && (DOSX & PHARLAP) 1243 if (writeInfo) 1244 { 1245 DWORD rAdr = realBase + (WORD)&PktReceiver; 1246 unsigned sel, ofs; 1247 1248 printf ("\nReceiver at %04X:%04X\n", RP_SEG(rAdr), RP_OFF(rAdr)); 1249 printf ("Realbase = %04X:%04X\n", RP_SEG(realBase),RP_OFF(realBase)); 1250 1251 sel = _FP_SEG (protBase); 1252 ofs = _FP_OFF (protBase); 1253 printf ("Protbase = %04X:%08X\n", sel,ofs); 1254 printf ("RealSeg = %04X\n", realSeg); 1255 1256 sel = _FP_SEG (rxOutOfsFp); 1257 ofs = _FP_OFF (rxOutOfsFp); 1258 printf ("rxOutOfsFp = %04X:%08X\n", sel,ofs); 1259 1260 sel = _FP_SEG (rxInOfsFp); 1261 ofs = _FP_OFF (rxInOfsFp); 1262 printf ("rxInOfsFp = %04X:%08X\n", sel,ofs); 1263 1264 printf ("Ready: *rxOutOfsFp = %04X *rxInOfsFp = %04X\n", 1265 *rxOutOfsFp, *rxInOfsFp); 1266 1267 PktQueueBusy (TRUE); 1268 printf ("Busy: *rxOutOfsFp = %04X *rxInOfsFp = %04X\n", 1269 *rxOutOfsFp, *rxInOfsFp); 1270 } 1271 #endif 1272 1273 memset (&pktStat, 0, sizeof(pktStat)); /* clear statistics */ 1274 PktQueueBusy (TRUE); 1275 return (TRUE); 1276 } 1277 1278 1279 /* 1280 * DPMI functions only for Watcom + DOS4GW extenders 1281 */ 1282 #if (DOSX & DOS4GW) 1283 LOCAL DWORD dpmi_get_real_vector (int intr) 1284 { 1285 union REGS r; 1286 1287 r.x.eax = 0x200; 1288 r.x.ebx = (DWORD) intr; 1289 int386 (0x31, &r, &r); 1290 return ((r.w.cx << 4) + r.w.dx); 1291 } 1292 1293 LOCAL WORD dpmi_real_malloc (int size, WORD *selector) 1294 { 1295 union REGS r; 1296 1297 r.x.eax = 0x0100; /* DPMI allocate DOS memory */ 1298 r.x.ebx = (size + 15) / 16; /* Number of paragraphs requested */ 1299 int386 (0x31, &r, &r); 1300 if (r.w.cflag & 1) 1301 return (0); 1302 1303 *selector = r.w.dx; 1304 return (r.w.ax); /* Return segment address */ 1305 } 1306 1307 LOCAL void dpmi_real_free (WORD selector) 1308 { 1309 union REGS r; 1310 1311 r.x.eax = 0x101; /* DPMI free DOS memory */ 1312 r.x.ebx = selector; /* Selector to free */ 1313 int386 (0x31, &r, &r); 1314 } 1315 #endif 1316 1317 1318 #if defined(DOSX) && (DOSX & PHARLAP) 1319 /* 1320 * Description: 1321 * This routine allocates conventional memory for the specified block 1322 * of code (which must be within the first 64K of the protected mode 1323 * program segment) and copies the code to it. 1324 * 1325 * The caller should free up the conventional memory block when it 1326 * is done with the conventional memory. 1327 * 1328 * NOTE THIS ROUTINE REQUIRES 386|DOS-EXTENDER 3.0 OR LATER. 1329 * 1330 * Calling arguments: 1331 * start_offs start of real mode code in program segment 1332 * end_offs 1 byte past end of real mode code in program segment 1333 * real_basep returned; real mode ptr to use as a base for the 1334 * real mode code (eg, to get the real mode FAR 1335 * addr of a function foo(), take 1336 * real_basep + (ULONG) foo). 1337 * This pointer is constructed such that 1338 * offsets within the real mode segment are 1339 * the same as the link-time offsets in the 1340 * protected mode program segment 1341 * prot_basep returned; prot mode ptr to use as a base for getting 1342 * to the conventional memory, also constructed 1343 * so that adding the prot mode offset of a 1344 * function or variable to the base gets you a 1345 * ptr to the function or variable in the 1346 * conventional memory block. 1347 * rmem_adrp returned; real mode para addr of allocated 1348 * conventional memory block, to be used to free 1349 * up the conventional memory when done. DO NOT 1350 * USE THIS TO CONSTRUCT A REAL MODE PTR, USE 1351 * REAL_BASEP INSTEAD SO THAT OFFSETS WORK OUT 1352 * CORRECTLY. 1353 * 1354 * Returned values: 1355 * 0 if error 1356 * 1 if success 1357 */ 1358 int RealCopy (ULONG start_offs, 1359 ULONG end_offs, 1360 REALPTR *real_basep, 1361 FARPTR *prot_basep, 1362 USHORT *rmem_adrp) 1363 { 1364 ULONG rm_base; /* base real mode para addr for accessing */ 1365 /* allocated conventional memory */ 1366 UCHAR *source; /* source pointer for copy */ 1367 FARPTR destin; /* destination pointer for copy */ 1368 ULONG len; /* number of bytes to copy */ 1369 ULONG temp; 1370 USHORT stemp; 1371 1372 /* First check for valid inputs 1373 */ 1374 if (start_offs >= end_offs || end_offs > 0x10000) 1375 return (FALSE); 1376 1377 /* Round start_offs down to a paragraph (16-byte) boundary so we can set up 1378 * the real mode pointer easily. Round up end_offs to make sure we allocate 1379 * enough paragraphs 1380 */ 1381 start_offs &= ~15; 1382 end_offs = (15 + (end_offs << 4)) >> 4; 1383 1384 /* Allocate the conventional memory for our real mode code. Remember to 1385 * round byte count UP to 16-byte paragraph size. We alloc it 1386 * above the DOS data buffer so both the DOS data buffer and the appl 1387 * conventional mem block can still be resized. 1388 * 1389 * First just try to alloc it; if we can't get it, shrink the appl mem 1390 * block down to the minimum, try to alloc the memory again, then grow the 1391 * appl mem block back to the maximum. (Don't try to shrink the DOS data 1392 * buffer to free conventional memory; it wouldn't be good for this routine 1393 * to have the possible side effect of making file I/O run slower.) 1394 */ 1395 len = ((end_offs - start_offs) + 15) >> 4; 1396 if (_dx_real_above(len, rmem_adrp, &stemp) != _DOSE_NONE) 1397 { 1398 if (_dx_cmem_usage(0, 0, &temp, &temp) != _DOSE_NONE) 1399 return (FALSE); 1400 1401 if (_dx_real_above(len, rmem_adrp, &stemp) != _DOSE_NONE) 1402 *rmem_adrp = 0; 1403 1404 if (_dx_cmem_usage(0, 1, &temp, &temp) != _DOSE_NONE) 1405 { 1406 if (*rmem_adrp != 0) 1407 _dx_real_free (*rmem_adrp); 1408 return (FALSE); 1409 } 1410 1411 if (*rmem_adrp == 0) 1412 return (FALSE); 1413 } 1414 1415 /* Construct real mode & protected mode pointers to access the allocated 1416 * memory. Note we know start_offs is aligned on a paragraph (16-byte) 1417 * boundary, because we rounded it down. 1418 * 1419 * We make the offsets come out rights by backing off the real mode selector 1420 * by start_offs. 1421 */ 1422 rm_base = ((ULONG) *rmem_adrp) - (start_offs >> 4); 1423 RP_SET (*real_basep, 0, rm_base); 1424 FP_SET (*prot_basep, rm_base << 4, SS_DOSMEM); 1425 1426 /* Copy the real mode code/data to the allocated memory 1427 */ 1428 source = (UCHAR *) start_offs; 1429 destin = *prot_basep; 1430 FP_SET (destin, FP_OFF(*prot_basep) + start_offs, FP_SEL(*prot_basep)); 1431 len = end_offs - start_offs; 1432 WriteFarMem (destin, source, len); 1433 1434 return (TRUE); 1435 } 1436 #endif /* DOSX && (DOSX & PHARLAP) */ 1437