Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1056 lines
35 KiB

  1. ;/*
  2. ; * Microsoft Confidential
  3. ; * Copyright (C) Microsoft Corporation 1988 - 1991
  4. ; * All Rights Reserved.
  5. ; */
  6. /* MEMBASE.C - MEM routines for determining and displaying memory usage
  7. * for conventional memory.
  8. */
  9. #include "stdio.h"
  10. #include "dos.h"
  11. #include "string.h"
  12. #include "stdlib.h"
  13. #include "msgdef.h"
  14. #include "version.h"
  15. #include "mem.h"
  16. /*���������������������������������������������������������������������������*/
  17. /* go through the arena and print out the program name,size etc for /P switch */
  18. /* but just keep storing them in a datastruct for later disp. for /C switch */
  19. unsigned int DisplayBaseDetail()
  20. {
  21. struct ARENA far *ThisArenaPtr;
  22. struct ARENA far *NextArenaPtr;
  23. struct ARENA far *ThisConfigArenaPtr;
  24. struct ARENA far *NextConfigArenaPtr;
  25. struct DEVICEHEADER far *ThisDeviceDriver;
  26. int SystemDataType;
  27. char SystemDataOwner[64];
  28. unsigned int far *UMB_Head_ptr;
  29. unsigned int long Out_Var1;
  30. unsigned int long Out_Var2;
  31. char Out_Str1[64];
  32. char Out_Str2[64];
  33. unsigned int msgno;
  34. InRegs.h.ah = (unsigned char) 0x52;
  35. intdosx(&InRegs,&OutRegs,&SegRegs);
  36. FP_SEG(SysVarsPtr) = FP_SEG(UMB_Head_ptr) = SegRegs.es;
  37. FP_OFF(SysVarsPtr) = OutRegs.x.bx;
  38. FP_OFF(UMB_Head_ptr) = 0x8c; /* ptr to UMB_HEAD in DOS Data */
  39. UMB_Head = *UMB_Head_ptr;
  40. if (!Classify)
  41. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  42. if (DataLevel > 0)
  43. {
  44. if (!Classify) {
  45. Sub0_Message(Title1Msg,STDOUT,Utility_Msg_Class);
  46. Sub0_Message(Title2Msg,STDOUT,Utility_Msg_Class);
  47. }
  48. }
  49. InRegs.h.ah = (unsigned char) 0x30;
  50. intdos(&InRegs, &OutRegs);
  51. if ( (OutRegs.h.al != (unsigned char) 3) || (OutRegs.h.ah < (unsigned char) 40) )
  52. UseArgvZero = TRUE;
  53. else UseArgvZero = FALSE;
  54. /* Display stuff below DOS */
  55. Out_Var1 = 0l;
  56. Out_Var2 = 0x400l;
  57. if (Classify) /* M003 */
  58. /* classify this memory also as part of DOS */
  59. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  60. else
  61. Sub4_Message(MainLineMsg,
  62. STDOUT,
  63. Utility_Msg_Class,
  64. &Out_Var1,
  65. BlankMsg,
  66. &Out_Var2,
  67. InterruptVectorMsg);
  68. Out_Var1 = 0x400l;
  69. Out_Var2 = 0x100l;
  70. if (Classify)
  71. /* classify this memory also as part of DOS */
  72. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  73. else
  74. Sub4_Message(MainLineMsg,
  75. STDOUT,
  76. Utility_Msg_Class,
  77. &Out_Var1,
  78. BlankMsg,
  79. &Out_Var2,
  80. ROMCommunicationAreaMsg);
  81. Out_Var1 = 0x500l;
  82. Out_Var2 = 0x200l;
  83. if (Classify)
  84. /* classify this memory also as part of DOS */
  85. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  86. else
  87. Sub4_Message(MainLineMsg,
  88. STDOUT,
  89. Utility_Msg_Class,
  90. &Out_Var1,
  91. BlankMsg,
  92. &Out_Var2,
  93. DOSCommunicationAreaMsg);
  94. /* Display the DOS data */
  95. /* Display the BIO data location and size */
  96. if (!Classify)
  97. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  98. Out_Var1 = 0x700l;
  99. Out_Var2 = (long) (FP_SEG(SysVarsPtr) - 0x70)*16l;
  100. if (Classify)
  101. /* classify this memory also as part of DOS */
  102. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  103. else
  104. Sub4_Message(MainLineMsg,
  105. STDOUT,
  106. Utility_Msg_Class,
  107. &Out_Var1,
  108. IbmbioMsg,
  109. &Out_Var2,
  110. SystemDataMsg);
  111. /* Display the Base Device Driver Locations and Sizes */
  112. /*********************************************************************/
  113. /* to do this get the starting address of the internal driver header */
  114. /* chain. Start from the first header and get the address of the */
  115. /* first header. Display the driver name and address by calling */
  116. /* "DISPLAYDEVICEDRIVER". Repeat this for next driver on the chain */
  117. /* until the last driver. Note that driver name is in the header. */
  118. /* The driver header addrs is in the system variable table from */
  119. /* INT 21H fun 52H call. */
  120. /*********************************************************************/
  121. BlockDeviceNumber = 0;
  122. for (ThisDeviceDriver = SysVarsPtr -> DeviceDriverChain;
  123. (FP_OFF(ThisDeviceDriver) != 0xFFFF);
  124. ThisDeviceDriver = ThisDeviceDriver -> NextDeviceHeader)
  125. { if ( FP_SEG(ThisDeviceDriver) < FP_SEG(SysVarsPtr) )
  126. DisplayDeviceDriver(ThisDeviceDriver,SystemDeviceDriverMsg);
  127. }
  128. /* Display the DOS data location and size */
  129. FP_SEG(ArenaHeadPtr) = FP_SEG(SysVarsPtr); /* ;an004; */
  130. FP_OFF(ArenaHeadPtr) = FP_OFF(SysVarsPtr) - 2; /* ;an004; */
  131. /* ;an004; */
  132. FP_SEG(ThisArenaPtr) = *ArenaHeadPtr; /* ;an004; */
  133. FP_OFF(ThisArenaPtr) = 0; /* ;an004; */
  134. if (!Classify)
  135. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  136. Out_Var1 = (long) FP_SEG(SysVarsPtr) * 16l;
  137. Out_Var2 = (long) ((AddressOf((char far *)ThisArenaPtr)) - Out_Var1); /* ;ac004; */
  138. if (Classify)
  139. /* classify this memory also as part of DOS */
  140. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  141. else
  142. Sub4_Message(MainLineMsg,
  143. STDOUT,
  144. Utility_Msg_Class,
  145. &Out_Var1,
  146. IbmdosMsg,
  147. &Out_Var2,
  148. SystemDataMsg);
  149. if (!Classify)
  150. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  151. /* Display the memory data */
  152. /* IO.SYS data area contains BUFFERS, FCBs, LAST DRIVE etc. They are contained */
  153. /* in a one huge memory block. This block has a seg iD 0008. This seg ID can */
  154. /* be found from the block header owner area. This seg id 0008:0000 */
  155. /* points to the buffer table as shown below. If seg id is 0008:000, then */
  156. /* using the seg id find the table. Each entry is contained in a sub block */
  157. /* within the main block. Each sub block has header and this header contains */
  158. /* id such as B for BUFFER, X for FCBs, I for IFS, D for external device */
  159. /* drivers. Go through the sub blocks and display the name ans size. that's all. */
  160. /* */
  161. /* If the block contains D, then it contains external drivers. The driver name */
  162. /* is not in the sub block. So we have to find the driver name from the driver */
  163. /* header chain. To do this get the address of the driver chain from syster */
  164. /* variable table from INT 21H FN 52H call. Go through the chain and findout */
  165. /* the name. Display name from the header and the size we got from the sub block. */
  166. /* */
  167. /* */
  168. /* After this main block, comes other buffer blocks which contains programs */
  169. /* such as command.com, doscolor, even MEM. From these blocks, get the program */
  170. /* name and the size and display them too. */
  171. /* */
  172. /* 0008:000->------------------ ------------------- */
  173. /* | BUFFERS | -------->|B (signature) | Block header */
  174. /* ------------------ ------------------- */
  175. /* | FCBs | -- | | */
  176. /* ------------------ | | Buffers data | */
  177. /* | IFSs | | | | */
  178. /* ------------------ | | | */
  179. /* | LAST DRIVE | | | | */
  180. /* ------------------ | -------------------- */
  181. /* | EXTERN DRIVER 1| | */
  182. /* ------------------ | ------------------- */
  183. /* | EXTERN DRIVER 2| | -------->|X (signature) | Block header */
  184. /* ------------------ ------------------- */
  185. /* | EXTERN DRIVER 3| | | */
  186. /* ------------------ | Buffers data | */
  187. /* | | */
  188. /* | | */
  189. /* | | */
  190. /* -------------------- */
  191. /* */
  192. /* For DOS 5.0, there are some additions to the above. Basically, we have
  193. /* three possible memory maps, to wit:
  194. /*
  195. /* DOS Loads Low DOS loads high
  196. /* 70:0 - BIOS data 70:0 - BIOS data
  197. /* DOS data DOS data
  198. /* BIOS + DOS code Sysinit data (arena name SD)
  199. /* (arena owner 8, name "SC") VDisk header (arena name SC)
  200. /* Sysinit data (arean owner 8, name SD)
  201. /*
  202. /* DOS tries to load high but fails
  203. /* 70:0 - BIOS data
  204. /* DOS data
  205. /* Sysinit data (arena name SD)
  206. /* DOS + BIOS code (arena name SC)
  207. /*
  208. /* We have to detect the special arena ownership marks and display them
  209. /* correctly. Everything after DOS and BIOS data should have an arena header
  210. /******************************************************************************/
  211. while (ThisArenaPtr -> Signature != (char) 'Z')
  212. {
  213. /* MSKK02 */
  214. #ifdef JAPAN
  215. if (ThisArenaPtr -> Owner == 8 || ThisArenaPtr -> Owner == 9 )
  216. #else
  217. if (ThisArenaPtr -> Owner == 8)
  218. #endif
  219. {
  220. FP_SEG(NextArenaPtr) = FP_SEG(ThisArenaPtr) + ThisArenaPtr -> Paragraphs + 1;
  221. FP_OFF(NextArenaPtr) = 0;
  222. Out_Var1 = AddressOf((char far *)ThisArenaPtr);
  223. Out_Var2 = (long) (ThisArenaPtr -> Paragraphs) * 16l;
  224. if (ThisArenaPtr->OwnerName[0] == 'S' &&
  225. ThisArenaPtr->OwnerName[1] == 'C')
  226. { /* display message for BIOS and DOS code */
  227. if (Classify)
  228. /* classify this memory also as part of DOS */
  229. { if (AddMem_to_PSP(8,Out_Var1,Out_Var2)) return(1); }
  230. else {
  231. msgno = (FP_SEG(ThisArenaPtr) < UMB_Head) ? IbmdosMsg:SystemMsg;
  232. Sub4_Message(MainLineMsg,
  233. STDOUT,
  234. Utility_Msg_Class,
  235. &Out_Var1,
  236. msgno,
  237. &Out_Var2,
  238. SystemProgramMsg);
  239. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  240. }
  241. }
  242. else /* display message for data */
  243. {
  244. if (!Classify)
  245. Sub4_Message(MainLineMsg,
  246. STDOUT,
  247. Utility_Msg_Class,
  248. &Out_Var1,
  249. #ifdef JAPAN
  250. (ThisArenaPtr -> Owner == 8) ? IbmbioMsg : AdddrvMsg,
  251. &Out_Var2,
  252. (ThisArenaPtr -> Owner == 8) ? SystemDataMsg : ProgramMsg );
  253. #else
  254. IbmbioMsg,
  255. &Out_Var2,
  256. SystemDataMsg);
  257. #endif
  258. FP_SEG(ThisConfigArenaPtr) = FP_SEG(ThisArenaPtr) + 1;
  259. FP_OFF(ThisConfigArenaPtr) = 0;
  260. while ( (FP_SEG(ThisConfigArenaPtr) > FP_SEG(ThisArenaPtr)) &&
  261. (FP_SEG(ThisConfigArenaPtr) < FP_SEG(NextArenaPtr)) )
  262. {
  263. strcpy(SystemDataOwner," ");
  264. switch(ThisConfigArenaPtr -> Signature)
  265. {
  266. case 'B':
  267. SystemDataType = ConfigBuffersMsg;
  268. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  269. break;
  270. case 'D':
  271. SystemDataType = ConfigDeviceMsg;
  272. if (AddMem_to_PSP(ThisConfigArenaPtr->Owner,((long)FP_SEG(ThisConfigArenaPtr)*16l) ,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  273. strcpy(SystemDataOwner,OwnerOf(ThisConfigArenaPtr));
  274. break;
  275. case 'F':
  276. SystemDataType = ConfigFilesMsg;
  277. if (AddMem_to_PSP(8,((long)FP_SEG(ThisConfigArenaPtr) *16l),((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  278. break;
  279. case 'I':
  280. SystemDataType = ConfigIFSMsg;
  281. strcpy(SystemDataOwner,OwnerOf(ThisConfigArenaPtr));
  282. if (AddMem_to_PSP(ThisConfigArenaPtr->Owner,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  283. break;
  284. case 'L':
  285. SystemDataType = ConfigLastDriveMsg;
  286. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  287. break;
  288. case 'S':
  289. SystemDataType = ConfigStacksMsg;
  290. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  291. break;
  292. case 'T': /* gga */
  293. SystemDataType = ConfigInstallMsg; /* gga */
  294. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  295. break; /* gga */
  296. case 'X':
  297. SystemDataType = ConfigFcbsMsg;
  298. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  299. break;
  300. // NTVDM for special kbd\mouse drivers
  301. case 'Q':
  302. SystemDataType = SystemProgramMsg;
  303. strcpy(SystemDataOwner,OwnerOf(ThisConfigArenaPtr));
  304. if (AddMem_to_PSP(ThisConfigArenaPtr->Owner,((long)FP_SEG(ThisConfigArenaPtr)*16l) ,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  305. break;
  306. #ifdef JAPAN
  307. case '?':
  308. SystemDataType = DataMsg;
  309. break;
  310. #endif
  311. default:
  312. SystemDataType = BlankMsg;
  313. if (AddMem_to_PSP(8,((long)ThisConfigArenaPtr) <<4,((long)ThisConfigArenaPtr->Paragraphs <<4) +1)) return(1);
  314. break;
  315. }
  316. /****************************************************/
  317. /* Found one, now display the owner name and size */
  318. /****************************************************/
  319. Out_Var1 = ((long) ThisConfigArenaPtr -> Paragraphs) * 16l;
  320. if (!Classify)
  321. Sub3_Message(DriverLineMsg,
  322. STDOUT,
  323. Utility_Msg_Class,
  324. SystemDataOwner,
  325. &Out_Var1,
  326. SystemDataType );
  327. NextConfigArenaPtr = ThisConfigArenaPtr;
  328. FP_SEG(NextConfigArenaPtr) += NextConfigArenaPtr -> Paragraphs + 1;
  329. if (ThisConfigArenaPtr -> Signature == (char) 'D')
  330. {
  331. FP_SEG(ThisDeviceDriver) = FP_SEG(ThisConfigArenaPtr) + 1;
  332. FP_OFF(ThisDeviceDriver) = 0;
  333. /* start MSKK bug fix - MSKK01 */
  334. while ( (FP_SEG(ThisDeviceDriver) > FP_SEG(ThisConfigArenaPtr)) &&
  335. (FP_SEG(ThisDeviceDriver) < FP_SEG(NextConfigArenaPtr)) ) {
  336. DisplayDeviceDriver(ThisDeviceDriver,InstalledDeviceDriverMsg);
  337. ThisDeviceDriver = ThisDeviceDriver -> NextDeviceHeader;
  338. }
  339. /* end MSKK bug fix - MSKK01 */
  340. }
  341. FP_SEG(ThisConfigArenaPtr) += ThisConfigArenaPtr -> Paragraphs + 1;
  342. }
  343. }
  344. }
  345. else {
  346. /*******************************************************************************/
  347. /* If not BIOS table, it is a program like MEM, etc. */
  348. /* calculate the size of the block occupied by the program and display program */
  349. /* name and size */
  350. /*******************************************************************************/
  351. Out_Var1 = AddressOf((char far *)ThisArenaPtr);
  352. Out_Var2 = ((long) (ThisArenaPtr -> Paragraphs)) * 16l;
  353. strcpy(Out_Str1,OwnerOf(ThisArenaPtr));
  354. strcpy(Out_Str2,TypeOf(ThisArenaPtr));
  355. if (Classify)
  356. { if (AddMem_to_PSP(ThisArenaPtr->Owner,Out_Var1,Out_Var2)) return(1); }
  357. else
  358. Sub4a_Message(MainLineMsg,
  359. STDOUT,
  360. Utility_Msg_Class,
  361. &Out_Var1,
  362. Out_Str1,
  363. &Out_Var2,
  364. Out_Str2);
  365. }
  366. FP_SEG(ThisArenaPtr) += ThisArenaPtr -> Paragraphs + 1;
  367. }
  368. Out_Var1 = AddressOf((char far *)ThisArenaPtr);
  369. Out_Var2 = ((long) (ThisArenaPtr -> Paragraphs)) * 16l;
  370. strcpy(Out_Str1,OwnerOf(ThisArenaPtr));
  371. strcpy(Out_Str2,TypeOf(ThisArenaPtr));
  372. if (Classify)
  373. { if (AddMem_to_PSP(ThisArenaPtr->Owner,Out_Var1,Out_Var2)) return(1); }
  374. else
  375. Sub4a_Message(MainLineMsg,
  376. STDOUT,
  377. Utility_Msg_Class,
  378. &Out_Var1,
  379. Out_Str1,
  380. &Out_Var2,
  381. Out_Str2);
  382. return(0); /* end of MEM main routine */
  383. }
  384. /*���������������������������������������������������������������������������*/
  385. void DisplayDeviceDriver(ThisDeviceDriver,DeviceDriverType)
  386. struct DEVICEHEADER far *ThisDeviceDriver;
  387. int DeviceDriverType;
  388. {
  389. char LocalDeviceName[16];
  390. int i;
  391. if (DataLevel < 2) return;
  392. if ( ((ThisDeviceDriver -> Attributes) & 0x8000 ) != 0 )
  393. { for (i = 0; i < 8; i++) LocalDeviceName[i] = ThisDeviceDriver -> Name[i];
  394. LocalDeviceName[8] = NUL;
  395. Sub2_Message(DeviceLineMsg,
  396. STDOUT,
  397. Utility_Msg_Class,
  398. LocalDeviceName,
  399. DeviceDriverType);
  400. }
  401. else {
  402. if ((int) ThisDeviceDriver -> Name[0] == 1)
  403. sprintf(&LocalDeviceName[0],SingleDrive,'A'+BlockDeviceNumber);
  404. else sprintf(&LocalDeviceName[0],MultipleDrives,
  405. 'A'+BlockDeviceNumber,
  406. 'A'+BlockDeviceNumber + ((int) ThisDeviceDriver -> Name[0]) - 1);
  407. Sub2_Message(DeviceLineMsg,
  408. STDOUT,
  409. Utility_Msg_Class,
  410. LocalDeviceName,
  411. DeviceDriverType);
  412. BlockDeviceNumber += (int) (ThisDeviceDriver -> Name[0]);
  413. }
  414. return;
  415. }
  416. /*���������������������������������������������������������������������������*/
  417. void DisplayBaseSummary()
  418. {
  419. struct PSP_STRUC
  420. {
  421. unsigned int int_20;
  422. unsigned int top_of_memory;
  423. };
  424. char far *CarvedPtr;
  425. unsigned long int total_mem; /* total memory in system */
  426. unsigned long int avail_mem; /* avail memory in system */
  427. unsigned long int free_mem; /* free memory */
  428. unsigned long biggest_free; /* largest free block now :M001 */
  429. struct PSP_STRUC far *PSPptr;
  430. /* skip a line */
  431. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  432. /* get PSP info */
  433. InRegs.h.ah = GET_PSP; /* get PSP function call */
  434. intdos(&InRegs,&OutRegs);
  435. FP_SEG(PSPptr) = OutRegs.x.bx; /* PSP segment */
  436. FP_OFF(PSPptr) = 0; /* offset 0 */
  437. /* Get total memory in system */
  438. int86(MEMORY_DET,&InRegs,&OutRegs);
  439. /* Convert to bytes */
  440. total_mem = (unsigned long int) OutRegs.x.ax * 1024l;
  441. avail_mem = total_mem;
  442. /* M004 BEGIN */
  443. /* Adjust for XBDA size */
  444. /* XBDA size should be added to total mem size reported by INT 12 */
  445. /* IFF XBDA is placed just at the end of conv.mem */
  446. /* IF EMM386 or QEMM is loaded, XBDA gets relocated to EMM driver mem */
  447. /* and int 12 reports correct size of memory in this case */
  448. InRegs.x.bx = 0;
  449. InRegs.x.ax = 0xc100;
  450. int86x(0x15, &InRegs, &OutRegs, &SegRegs);
  451. if (OutRegs.x.cflag == 0)
  452. {
  453. if (total_mem == ((unsigned long)SegRegs.es) * 16ul) {
  454. FP_SEG(CarvedPtr) = SegRegs.es;
  455. FP_OFF(CarvedPtr) = 0;
  456. total_mem = total_mem + ( (unsigned long int) (*CarvedPtr) * 1024l) ; /* ;an002; dms;adjust total for */
  457. }
  458. } /* RAM carve value */
  459. /* M004 END */
  460. Sub1_Message(TotalMemoryMsg,STDOUT,Utility_Msg_Class,&total_mem);
  461. Sub1_Message(AvailableMemoryMsg,STDOUT,Utility_Msg_Class,&avail_mem);
  462. /* Calculate the total memory used. PSP segment * 16. Subtract from total to
  463. get free_mem */
  464. free_mem = (DOS_TopOfMemory * 16l) - (FP_SEG(PSPptr)*16l); /* ;an000;ac005; */
  465. /* Get largest free block in system :M001 */
  466. InRegs.x.ax = 0x4800; /* M001 */
  467. InRegs.x.bx = 0xffff; /* M001 */
  468. intdos(&InRegs, &OutRegs); /* M001 */
  469. biggest_free = OutRegs.x.bx * 16L; /* Size of largest block now :M001 */
  470. /* The largest free block in the system is either the block we are
  471. currently in or the block we have allocated. We can either be the
  472. topmost program or be loaded in a hole or UMB. In either case, the
  473. larger of the 2 values gives us the largest free block :M001
  474. */
  475. if ( biggest_free > free_mem ) /* M001 */
  476. free_mem = biggest_free; /* M001 */
  477. Sub1_Message(FreeMemoryMsg,STDOUT,Utility_Msg_Class,&free_mem);
  478. return;
  479. } /* end of display_low_total */
  480. /*���������������������������������������������������������������������������*/
  481. char *OwnerOf(ArenaPtr)
  482. struct ARENA far *ArenaPtr;
  483. {
  484. char far *StringPtr;
  485. char *o;
  486. unsigned far *EnvironmentSegmentPtr;
  487. unsigned PspSegment;
  488. int i,fPrintable;
  489. o = &OwnerName[0];
  490. *o = NUL;
  491. sprintf(o,UnOwned);
  492. PspSegment = ArenaPtr -> Owner;
  493. if (PspSegment == 0) sprintf(o,Ibmdos);
  494. else if (PspSegment == 8) sprintf(o,Ibmbio);
  495. else {
  496. FP_SEG(ArenaPtr) = PspSegment-1; /* -1 'cause Arena is 16 bytes before PSP */
  497. StringPtr = (char far *) &(ArenaPtr -> OwnerName[0]);
  498. /* M002 BEGIN */
  499. fPrintable = TRUE;
  500. /* Chars below 0x20 (Space) and char 0x7f are not printable in US and
  501. * European Code pages. The following code checks for it and does not print
  502. * such names. - Nagara 11/20/90
  503. */
  504. for (i = 0; i < 8;i++,StringPtr++) {
  505. #ifdef DBCS
  506. if ( ((unsigned char)*StringPtr < 0x20) | ((unsigned char)*StringPtr == 0x7f) ) {
  507. #else
  508. if ( (*StringPtr < 0x20) | (*StringPtr == 0x7f) ) {
  509. #endif
  510. /* unprintable char ? */
  511. if (*StringPtr) fPrintable = FALSE;
  512. break;
  513. }
  514. }
  515. if (fPrintable) { /* the name is printable */
  516. StringPtr = (char far *) &(ArenaPtr -> OwnerName[0]);
  517. for (i = 0; i < 8;i++)
  518. *o++ = *StringPtr++;
  519. *o = (char) '\0';
  520. }
  521. /* M002 END */
  522. }
  523. if (UseArgvZero) GetFromArgvZero(PspSegment,EnvironmentSegmentPtr);
  524. return(&OwnerName[0]);
  525. }
  526. /*���������������������������������������������������������������������������*/
  527. void GetFromArgvZero(PspSegment,EnvironmentSegmentPtr)
  528. unsigned PspSegment;
  529. unsigned far *EnvironmentSegmentPtr;
  530. {
  531. char far *StringPtr;
  532. char *OutputPtr;
  533. unsigned far *WordPtr;
  534. OutputPtr = &OwnerName[0];
  535. if (UseArgvZero)
  536. {
  537. if (PspSegment < FP_SEG(ArenaHeadPtr))
  538. {
  539. if (*OutputPtr == NUL) sprintf(OutputPtr,Ibmdos);
  540. }
  541. else {
  542. FP_SEG(EnvironmentSegmentPtr) = PspSegment;
  543. FP_OFF(EnvironmentSegmentPtr) = 44;
  544. /* FP_SEG(StringPtr) = *EnvironmentSegmentPtr; */
  545. FP_SEG(StringPtr) = FP_SEG(EnvironmentSegmentPtr);
  546. FP_OFF(StringPtr) = 0;
  547. while ( (*StringPtr != NUL) || (*(StringPtr+1) != NUL) ) StringPtr++;
  548. StringPtr += 2;
  549. WordPtr = (unsigned far *) StringPtr;
  550. if (*WordPtr == 1)
  551. {
  552. StringPtr += 2;
  553. while (*StringPtr != NUL)
  554. *OutputPtr++ = *StringPtr++;
  555. *OutputPtr++ = NUL;
  556. while ( OutputPtr > &OwnerName[0] )
  557. { if (*OutputPtr == (char) '.') *OutputPtr = NUL;
  558. if ( (*OutputPtr == (char) '\\') || (*OutputPtr == (char) ':') )
  559. { OutputPtr++;
  560. break;
  561. }
  562. OutputPtr--;
  563. }
  564. }
  565. }
  566. }
  567. strcpy(&OwnerName[0],OutputPtr);
  568. return;
  569. }
  570. /*���������������������������������������������������������������������������*/
  571. char *TypeOf(Header)
  572. struct ARENA far *Header;
  573. {
  574. char *t;
  575. unsigned PspSegment;
  576. unsigned far *EnvironmentSegmentPtr;
  577. unsigned int Message_Number;
  578. char far *Message_Buf;
  579. unsigned int i;
  580. t = &TypeText[0];
  581. *t = NUL;
  582. Message_Number = 0xff; /* ;an000; initialize number value */
  583. if (Header -> Owner == 8) Message_Number = StackMsg;
  584. if (Header -> Owner == 0) Message_Number = FreeMsg;
  585. PspSegment = Header -> Owner;
  586. if (PspSegment < FP_SEG(ArenaHeadPtr))
  587. {
  588. if (Message_Number == 0xff) Message_Number = BlankMsg;
  589. }
  590. else {
  591. FP_SEG(EnvironmentSegmentPtr) = PspSegment;
  592. FP_OFF(EnvironmentSegmentPtr) = 44;
  593. if (PspSegment == FP_SEG(Header)+1)
  594. Message_Number = ProgramMsg;
  595. else if ( *EnvironmentSegmentPtr == FP_SEG(Header)+1 )
  596. Message_Number = EnvironMsg;
  597. else
  598. Message_Number = DataMsg;
  599. }
  600. InRegs.x.ax = Message_Number;
  601. InRegs.h.dh = Utility_Msg_Class;
  602. sysgetmsg(&InRegs,&SegRegs,&OutRegs);
  603. FP_OFF(Message_Buf) = OutRegs.x.si;
  604. FP_SEG(Message_Buf) = SegRegs.ds;
  605. i = 0;
  606. while ( *Message_Buf != (char) '\x0' )
  607. TypeText[i++] = *Message_Buf++;
  608. TypeText[i++] = '\x0';
  609. return(t);
  610. }
  611. /* M003 BEGIN */
  612. /*----------------------------------------------------------------------*/
  613. /* AddMem_to_PSP */
  614. /* Entry: PSP_ADDR (to which this mem. should be added) */
  615. /* ARENA_START_ADDR */
  616. /* Length_of_Arena */
  617. /* Exit: mem_table updated. */
  618. /* returns 1 if more than MAX_CL_ENTRIES in mem_table */
  619. /* else 0 */
  620. /* */
  621. /* CAVEATS: */
  622. /* -------- */
  623. /* 1. any system area (BIOS,SYSINIT,DOS ) code/data is listed as belonging */
  624. /* to PSP 8. */
  625. /* */
  626. /* 2. We look at the UMB_HEAD in DOS DATA to determine whether an arena */
  627. /* is in UMB or not; For the Arena at the UMB boundary, we add one */
  628. /* para to conv. and remaining to UMB portion of that PSP */
  629. /* */
  630. /* 3. Any free memory is always added as a new entry in the mem_table */
  631. /* instead of just adding the sizes to an existing FREE entry */
  632. /* Free memory gets added to the previous free memory if they are */
  633. /* contiguous */
  634. /* */
  635. /* 4. The no of programs/free arenas cannot exceed a max of (100) */
  636. /* (defined by MAX_CLDATA_INDEX ) */
  637. /* If the memory is fragmented and a lot of small TSRs loaded such */
  638. /* that we exceed this limit, we TERMINATE */
  639. /* */
  640. /* 5. Mem occupied by this MEM are also reported as FREE mem */
  641. /* */
  642. /*----------------------------------------------------------------------*/
  643. unsigned int AddMem_to_PSP(psp,start_addr,length)
  644. unsigned int psp;
  645. unsigned long start_addr,length;
  646. {
  647. unsigned int para_no,len_in_paras,CurPSP;
  648. int i;
  649. extern unsigned int _psp;
  650. para_no = (unsigned int)(start_addr >> 4); /* convert to paras*/
  651. len_in_paras = (unsigned int)(length >> 4); /* convert to paras */
  652. CurPSP = psp;
  653. if (psp == _psp) psp = 0; /* treat MEM's arenas as FREE */
  654. if (!psp) {
  655. if (LastPSP == _psp) { /* if the prev.arena was MEM */
  656. i = noof_progs -1; /* look at the last entry */
  657. if (mem_table[i].psp_add != psp) /* was the last entry free ?*/
  658. i++;
  659. else len_in_paras++; /* account for one free arena header */
  660. }
  661. else i = noof_progs; /* new entry for FREE mem */
  662. }
  663. else
  664. for (i = 0;i < noof_progs;i++)
  665. if (mem_table[i].psp_add == psp) break;
  666. /* if psp is not already listed in the table, add it */
  667. if (i == noof_progs) {
  668. if (noof_progs == MAX_CLDATA_INDEX) {
  669. /* use parse error message proc to display err msg */
  670. Parse_Message(CMemFragMsg,STDERR,Utility_Msg_Class,(char far *) NULL);
  671. return(1);
  672. }
  673. mem_table[i].psp_add = psp;
  674. noof_progs++;
  675. }
  676. /* add the memory to the table entry */
  677. if (para_no < UMB_Head)
  678. mem_table[i].mem_conv += len_in_paras;
  679. else if (para_no == UMB_Head) {
  680. mem_table[i].mem_conv++;
  681. mem_table[i].mem_umb = len_in_paras-1;
  682. }
  683. else mem_table[i].mem_umb += len_in_paras;
  684. LastPSP = CurPSP;
  685. return(0);
  686. }
  687. /*���������������������������������������������������������������������������*/
  688. /************************************************************************/
  689. /* DisplayClassification */
  690. /* Main display proc for /C switch */
  691. /* */
  692. /* ENTRY: none */
  693. /* */
  694. /* EXIT: none */
  695. /* */
  696. /* find out if UMB is available by going through mem_table entries */
  697. /* (also find out MEM's size from these entries) */
  698. /* display memory break up for conventional memory */
  699. /* if (UMB in system) display memmory break up for UMB memory */
  700. /* display the total free size (= total free in conv.+total_free */
  701. /* in UMB + MEM's size ) */
  702. /* call DispBigFree to display the largest prog. sizes in Conv.&UMB*/
  703. /* */
  704. /************************************************************************/
  705. #define CONVONLY 0
  706. #define UMBONLY 1
  707. void DisplayClassification()
  708. {
  709. unsigned long tot_freemem=0L;
  710. char ShDSizeName[12];
  711. int i;
  712. unsigned int cur_psp;
  713. char fUMBAvail=0;
  714. /* get PSP info */
  715. InRegs.h.ah = GET_PSP; /* get PSP function call */
  716. intdos(&InRegs,&OutRegs);
  717. cur_psp = OutRegs.x.bx; /* psp of MEM */
  718. for (i=0;i <noof_progs;i++) {
  719. if (mem_table[i].mem_umb)
  720. fUMBAvail = TRUE;
  721. if (mem_table[i].psp_add == cur_psp) {
  722. tot_freemem += (long)(mem_table[i].mem_conv + mem_table[i].mem_umb);
  723. if (fUMBAvail) break;
  724. }
  725. }
  726. tot_freemem *=16l; /* convert to bytes */
  727. Sub0_Message(CTtlConvMsg,STDOUT,Utility_Msg_Class);
  728. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  729. tot_freemem += DispMemClass(CONVONLY);
  730. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  731. if (fUMBAvail) {
  732. Sub0_Message(CTtlUMBMsg,STDOUT,Utility_Msg_Class);
  733. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  734. tot_freemem += DispMemClass(UMBONLY);
  735. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  736. }
  737. sprintf(ShDSizeName," (%5.1fK)",((float)tot_freemem)/1024l );
  738. i = (fUMBAvail) ? CSumm1Msg:CSumm1AMsg;
  739. SubC2_Message(i,STDOUT,&tot_freemem, ShDSizeName);
  740. DispBigFree(fUMBAvail,cur_psp);
  741. }
  742. /************************************************************************/
  743. /* DispMemClass */
  744. /* (displays the progs and FREE sizes in either the */
  745. /* conventional or UMB) */
  746. /* */
  747. /* ENTRY: memtype => Conventional(0) /UMB (1) */
  748. /* */
  749. /* EXIT: total_freemem_size (in bytes) */
  750. /* */
  751. /* go through mem_table entries and display NON-ZERO size, NON-FREE */
  752. /* entries in the given mem_type */
  753. /* go through mem_table entries and display NON_ZERO size, FREE */
  754. /* entries in the given mem_type */
  755. /* calculate the total free mem size in the given mem type &return */
  756. /* */
  757. /* CAVEATS: */
  758. /* Arenas marked as belonging to MSDOS (code) in UMB are displayed */
  759. /* as SYSTEM */
  760. /* */
  761. /************************************************************************/
  762. unsigned long DispMemClass(memtype)
  763. int memtype;
  764. {
  765. int i,msgtype;
  766. unsigned int cur_memsize;
  767. unsigned long memsize;
  768. char *nameptr;
  769. char ShDSizeName[12];
  770. unsigned long tot_free = 0;
  771. struct ARENA far *ArenaPtr;
  772. Sub0_Message(CTtlNameMsg,STDOUT,Utility_Msg_Class);
  773. Sub0_Message(CTtlUScoreMsg,STDOUT,Utility_Msg_Class);
  774. for (i=0; i <noof_progs; i++) {
  775. cur_memsize = (memtype == CONVONLY) ? mem_table[i].mem_conv:mem_table[i].mem_umb;
  776. if (!cur_memsize) continue;
  777. if (!mem_table[i].psp_add) continue;
  778. msgtype = 0;
  779. if (mem_table[i].psp_add == 8) /* if DOS area */
  780. msgtype = (memtype == CONVONLY)?IbmdosMsg:SystemMsg;
  781. if (!msgtype) {
  782. FP_SEG(ArenaPtr) = mem_table[i].psp_add-1;
  783. /* -1 'cause Arena is 16 bytes before PSP */
  784. FP_OFF(ArenaPtr) = 0;
  785. nameptr = OwnerOf(ArenaPtr);
  786. }
  787. else nameptr = NULL;
  788. memsize = ((long) cur_memsize) *16l;
  789. sprintf(ShDSizeName," (%5.1fK)",((float)memsize)/1024l );
  790. SubC4_Message(MainLineMsg,STDOUT,nameptr,msgtype,&memsize, ShDSizeName);
  791. }
  792. for (i=0; i <noof_progs; i++) {
  793. if (mem_table[i].psp_add) continue;
  794. cur_memsize = (memtype == CONVONLY) ? mem_table[i].mem_conv:mem_table[i].mem_umb;
  795. if (!cur_memsize) continue;
  796. tot_free += (long) cur_memsize;
  797. memsize = ((long) cur_memsize) *16l;
  798. sprintf(ShDSizeName," (%5.1fK)",((float)memsize)/1024l );
  799. SubC4_Message(MainLineMsg,STDOUT,NULL,CFreeMsg,&memsize, ShDSizeName);
  800. }
  801. tot_free *= 16l;
  802. sprintf(ShDSizeName," (%5.1fK)",((float)tot_free )/1024l );
  803. Sub0_Message(NewLineMsg,STDOUT,Utility_Msg_Class);
  804. SubC2_Message(CTotalFreeMsg,STDOUT,&tot_free, ShDSizeName);
  805. return(tot_free);
  806. }
  807. /************************************************************************/
  808. /* DispBigFree */
  809. /* ENTRY: UMBAvailable? (flag) 1-> there is UMB */
  810. /* MEM's PSP */
  811. /* EXIT: none (largest prog.sizes displayed) */
  812. /* This finds out and displays the largest contig.mem available in */
  813. /* in Conventional and UMB memory */
  814. /* This calculation is done assuming that MEM is not loaded */
  815. /* */
  816. /************************************************************************/
  817. void DispBigFree(fUMBAvail,cur_psp)
  818. char fUMBAvail;
  819. unsigned int cur_psp;
  820. {
  821. int i;
  822. unsigned int ConvBigFree = 0;
  823. unsigned int UMBBigFree = 0;
  824. unsigned long TmpBigFree;
  825. char ShDSizeName[12];
  826. unsigned far *Env_ptr;
  827. unsigned int env_mem,BigFree;
  828. char fMEMHigh;
  829. /* assume that the biggest free size is the top of mem we got when */
  830. /* MEM was loaded */
  831. FP_SEG(Env_ptr) = cur_psp;
  832. FP_OFF(Env_ptr) = 44; /* get the env for MEM */
  833. FP_SEG(Env_ptr) = (*Env_ptr)-1; /* get to arena for env. */
  834. FP_OFF(Env_ptr) = 3; /* get the size of environment */
  835. env_mem = *Env_ptr + 1; /* 1 extra para for arena header */
  836. fMEMHigh = (char)((cur_psp > UMB_Head) ? 1:0);
  837. BigFree = DOS_TopOfMemory - cur_psp;
  838. if (fMEMHigh ) /* mem was loaded higher */
  839. UMBBigFree = BigFree;
  840. else
  841. ConvBigFree = BigFree;
  842. for (i =0; i<noof_progs;i++) {
  843. if (mem_table[i].psp_add) continue; /* skip non-FREE entries */
  844. if (mem_table[i].mem_conv > ConvBigFree)
  845. ConvBigFree = mem_table[i].mem_conv;
  846. if (mem_table[i].mem_umb > UMBBigFree)
  847. UMBBigFree = mem_table[i].mem_umb;
  848. }
  849. if (fMEMHigh) { /* MEM was loaded high */
  850. if (FP_SEG(Env_ptr) > UMB_Head) /* env also in UMB */
  851. if (UMBBigFree == (BigFree + env_mem))
  852. UMBBigFree = BigFree;
  853. }
  854. else { /* MEM was loaded low */
  855. if (FP_SEG(Env_ptr) < UMB_Head) /* env also in Conv */
  856. if (ConvBigFree == (BigFree + env_mem))
  857. ConvBigFree = BigFree;
  858. }
  859. TmpBigFree = ((unsigned long)ConvBigFree) * 16l;
  860. sprintf(ShDSizeName," (%5.1fK)",((float)TmpBigFree)/1024l );
  861. SubC2_Message(CSumm2Msg,STDOUT,&TmpBigFree, ShDSizeName);
  862. if (fUMBAvail) {
  863. TmpBigFree = ((unsigned long)UMBBigFree) * 16l;
  864. sprintf(ShDSizeName," (%5.1fK)",((float)TmpBigFree)/1024l );
  865. SubC2_Message(CSumm3Msg,STDOUT,&TmpBigFree, ShDSizeName);
  866. }
  867. }
  868. /* M003 END */
  869.