On the DOS platform, you can use the following methods to operate the serial port: BIOS call, hardware interruption through the serial port, or round-robin of the serial port hardware, this chapter will introduce and give examples of the above three methods. 1. BIOS interruption In the DOS Operating System, ibm pc and its compatible devices provide a flexible serial port I/O access method, that is, the ROM BIOS serial communication routine is called through int 14h. When Ah is set to a different value, different functions are generated: AH 0 initialization Port Ah 1 write characters to the serial port AH 2 read characters from the serial port Ah 3 fetch communication port status When initializing the port (when Ah = 0), you need to assign a byte of initialization parameter to the Al register. Meaning 1;
Figure 1 Al register settings when int 14h is called |
When writing characters to the serial port (when Ah = 1), the characters in the Al register are the characters to be written; When writing characters to the serial port (when Ah = 2), the characters in the Al register are the characters to be read. Take a look at the following routine:
# Include <stdio. h> # Include <dos. h> # Include <bios. h> # Define Str "Author: SBH" Union regs inregs, outregs;Main () { // Set serial port Parameters Init_rs232 (); // Example of serial port writing Write_rs232 (STR, strlen (STR )); // Example of reading the serial port Read_rs232 (); Return (0 ); } Init_rs232 () { Do { Inregs. H. Ah = 0; // Ah = 0 indicates the initialization port. Inregs. H. Al = 0xe7; Inregs. X. dx = 0; // COM1 Int86 (0x14, & inregs, & outregs ); } While (outregs. H. Ah> = 0x80 ); Return (0 ); } Write_rs232 (char * string, int Len) { Int I; Do { Inregs. H. Ah = 1; // send the character of the Al register Inregs. H. Al = * string; Inregs. X. dx = 0; Int86 (0x14, & inregs, & outregs ); } While (outregs. H. Al> = 0x80 ); For (I = 1; I <Len; I ++) { Inregs. H. Ah = 1; Inregs. H. Al = * (string + I ); Inregs. X. dx = 0; Int86 (0x14, & inregs, & outregs ); } } Read_rs232 () { Do { Inregs. H. Ah = 2; // read characters in the Al register Inregs. X. dx = 0; Int86 (0x14, & inregs, & outregs ); } While (outregs. H. Al! = 3 | outregs. H. Ah> = 0x80 ); Return (0 ); } |
The prototype of the int86 function used is:
Int _ cdecl int86 (INT intno, Union regs * inregs, Union regs * outregs ); |
The int86 () function can call the BIOS function. Currently, programmers have little access to this function, and 80% of programmers have never even seen this function. In fact, in the era of DOS, int86 () is almost one of the most common and core functions. It can be said that in that era, no int86 () means no programming. In combination with int86, a consortium such as regs is defined:
Union regs { Struct wordregs X; Struct byteregs h; }; |
The wordregs is defined:
Struct wordregs { Unsigned int ax, BX, CX, dx, Si, Di, Cflag/* Carry flag */, Flags/* sign register */; }; |
Byteregs is defined:
Struct byteregs { Unsigned char Al, ah, BL, BH, Cl, CH, DL, DH; }; |
It turns out that wordregs and byteregs are 16-bit registers in the 8086 processor! Therefore, when the CPU reaches 286 and 386, installing DOS is also based on the CPU usage mode! Another function is similar to int86:
Int _ cdecl int86x (INT intno, Union regs inregs, Union regs outregs, struct sregs segregs ); |
The sregs is a segment register struct, defined:
Struct sregs { Unsigned int es; Unsigned int Cs; Unsigned int SS; Unsigned int Ds; }; |
Both int86 and int86x functions execute a 8086 Soft Interrupt specified by the intno parameter. Before the Soft Interrupt is executed, both functions place the content in inregs into each register (int86x also places segregs. x. es and segregs. x. DS value is stored in the corresponding segment register). After Soft Interrupt is returned, both functions store the value of the current Register to outregs, and copy the system carry sign to outregs. s. in cflag, store the 8086 flag register value to outregs. x. flag (int86x also restores DS and sets segregs. es and segregs. DS value is the value of the corresponding segment register ). Refer to the BIOS interrupt call manual and find that most calls do not use the ES and DS registers. Therefore, the int86 function is often used in programming. 2. Hardware interruption To give readers an intuitive impression, we can view the resource properties of COM in the Windows operating system to get the interrupt number of a COM, 2 (enabled in the Device Manager in this dialog box ).
Figure 2 com interrupt number |
In fact, com directly corresponds to an interrupt, and the system also allocates a fixed interrupt number for various hardware according to certain rules, such as table 1. Table 1 interrupted vector table
INT (HEX) |
IRQ |
Common uses |
08 |
0 |
System Timer |
09 |
1 |
Keyboard |
0a |
2 |
Redirected |
0b |
3 |
Serial comms. com2/com4 |
0c |
4 |
Serial comms. COM1/com3 |
0d |
5 |
Reserved/sound card |
0e |
6 |
Floppy disk Controller |
0f |
7 |
Parallel comms. |
70 |
8 |
Real Time Clock |
71 |
9 |
Reserved |
72 |
10 |
Reserved |
73 |
11 |
Reserved |
74 |
12 |
PS/2 mouse |
75 |
13 |
Maths co-Processor |
76 |
14 |
Hard Disk Drive |
77 |
15 |
Reserved |
By compiling the interrupt service program corresponding to com, we can also operate the serial port. Related functions include: (1) set the interrupt vector table
/* Dos. H */ Void _ cdecl setvect (INT interruptno, void interrupt (* ISR )()); |
For example, if the interrupt Number of com3 is 4, the address of the corresponding interrupt vector table is 0x0c, and the function of the interrupt program corresponding to 0x0c is set:
Setvect (0x0c, port1int ); |
The interrupt service program port1int is:
Void interrupt port1int () { Int C; Do { C = inportb (port1 + 5 ); If (C & 1) { Buffer [Bufferin] = inportb (port1 ); Bufferin ++; If (Bufferin = 1024) Bufferin = 0; } } While (C & 1 ); Outportb (0x20, 0x20 ); } |
The above interrupt service program checks whether any character can be received, and then reads it from the UART through the inportb (port1) Statement and puts it into the input buffer. Continuously checks UART to read all available data in one interruption. The final "outportb (0x20, 0x20);" statement tells the Programmable Interrupt Controller (PIC) that the interrupt has been completed. (2) read the interrupted vector table
/* Dos. H */ Void interrupt (* _ cdecl getvect (INT interruptno ))(); |
For example:
Oldport1isr = getvect (intvect ); |
The oldport1isr is defined:
Void interrupt (* oldport1isr )(); |
We integrate the setvect () function, interrupt service program, and getvect () function to provide a complete routine compiled by Craig peaco CK:
/* Name: Sample comm's Program-1024 byte buffer-buff1024.c */ /* Written by: Craig peacock <cpeacock@senet.com.au> */ # Include <dos. h> # Include <stdio. h> # Include <conio. h># Define port1 0x3f8/* port address goes here */ # Define intvect 0x0c/* COM port's IRQ here (must also change PIC setting )*/ /* Defines serial ports base address */ /* COM1 0x3f8 */ /* Com2 0x2f8 */ /* Com3 0x3e8 */ /* Com4 0x2e8 */ Int Bufferin = 0; Int bufferout = 0; Char ch; Char buffer [1025]; Void interrupt (* oldport1isr )(); Void interrupt port1int ()/* interrupt service routine (ISR) for port1 */ { Int C; Do { C = inportb (port1 + 5 ); If (C & 1) { Buffer [Bufferin] = inportb (port1 ); Bufferin ++; If (Bufferin = 1024) { Bufferin = 0; } } } While (C & 1 ); Outportb (0x20, 0x20 ); } Void main (void) { Int C; Outportb (port1 + 1, 0);/* Turn off interrupts-port1 */ Oldport1isr = getvect (intvect);/* save old interrupt vector of later Recovery */ Setvect (intvect, port1int);/* Set interrupt vector entry */ /* COM1-0x0c */ /* Com2-0x0b */ /* Com3-0x0c */ /* Com4-0x0b */ /* Port 1-Communication Settings */ Outportb (port1 + 3, 0x80);/* Set dlab on */ Outportb (port1 + 0, 0x0c);/* Set baud rate-divisor latch low byte */ /* Default 0 x 03= 38,400 BPS */ /* 0x01 = 115,200 BPS */ /* 0x02 = 57,600 BPS */ /* 0x06 = 19,200 BPS */ /* 0x0c = 9,600 BPS */ /* 0x18 = 4,800 BPS */ /* 0x30 = 2,400 BPS */ Outportb (port1 + 1, 0x00);/* Set baud rate-divisor latch high byte */ Outportb (port1 + 3, 0x03);/* 8 bits, no parity, 1 stop bit */ Outportb (port1 + 2, 0xc7);/* FIFO control register */ Outportb (port1 + 4, 0x0b);/* Turn on DTR, RTS, and out2 */ Outportb (0x21, (inportb (0x21) & 0xef);/* Set Programmable Interrupt Controller */ /* COM1 (irq4)-0xef */ /* Com2 (irq3)-0xf7 */ /* Com3 (irq4)-0xef */ /* Com4 (irq3)-0xf7 */ Outportb (port1 + 1, 0x01);/* interrupt when data has ed */ Printf ("/nsample comm's program. Press ESC to quit/N "); Do { If (Bufferin! = Bufferout) { Ch = buffer [bufferout]; Bufferout ++; If (bufferout = 1024) { Bufferout = 0; } Printf ("% C", CH ); } If (kbhit ()) { C = getch (); Outportb (port1, C ); } } While (C! = 27 ); Outportb (port1 + 1, 0 ); /* Turn off interrupts-port1 */ Outportb (0x21, (inportb (0x21) | 0x10);/* mask IRQ Using PIC */ /* COM1 (irq4)-0x10 */ /* Com2 (irq3)-0x08 */ /* Com3 (irq4)-0x10 */ /* Com4 (irq3)-0x08 */ Setvect (intvect, oldport1isr);/* restore old interrupt vector */ } |
3. Hardware Query By reading and writing the hardware port corresponding to the serial UART, we can control the serial port sending and receiving. See the following example:
/* Name: Sample comm's Program-polled version-termpoll. C */ /* Written by: Craig peacock <cpeacock@senet.com.au> */ # Include <dos. h> # Include <stdio. h> # Include <conio. h>00000000000000000000 # define port1 0x3f8 /* Defines serial ports base address */ /* COM1 0x3f8 */ /* Com2 0x2f8 */ /* Com3 0x3e8 */ /* Com4 0x2e8 */ Void main (void) { Int C; Int ch; Outportb (port1 + 1, 0);/* Turn off interrupts-port1 */ /* Port 1-Communication Settings */ Outportb (port1 + 3, 0x80);/* Set dlab on */ Outportb (port1 + 0, 0x03);/* Set baud rate-divisor latch low byte */ /* Default 0 x 03= 38,400 BPS */ /* 0x01 = 115,200 BPS */ /* 0x02 = 57,600 BPS */ /* 0x06 = 19,200 BPS */ /* 0x0c = 9,600 BPS */ /* 0x18 = 4,800 BPS */ /* 0x30 = 2,400 BPS */ Outportb (port1 + 1, 0x00);/* Set baud rate-divisor latch high byte */ Outportb (port1 + 3, 0x03);/* 8 bits, no parity, 1 stop bit */ Outportb (port1 + 2, 0xc7);/* FIFO control register */ Outportb (port1 + 4, 0x0b);/* Turn on DTR, RTS, and out2 */ Printf ("/nsample comm's program. Press ESC to quit/N "); Do { C = inportb (port1 + 5);/* check to see if Char has been */ /* Received .*/ If (C & 1) { Ch = inportb (port1);/* if so, then get char */ Printf ("% C", CH ); }/* Print Char to screen */ If (kbhit ()) { Ch = getch ();/* If key pressed, get char */ Outportb (port1, CH ); }/* Send Char to serial port */ } While (Ch! = 27);/* quit when ESC (ASC 27) is pressed */ } |
In the program
C = inportb (port1 + 5);/* check to see if Char has been */ /* Received .*/ If (C & 1) |
Check the port1 + 5 port address and use C & 1 to determine whether data is received by UART. The port range corresponding to UART can be intuitively seen in figure 2. |