# $\begin{array}{c} \mathbf{GMRT} \\ \mathbf{MONITOR} \ \mathbf{AND} \ \mathbf{CONTROL} \ \mathbf{MODULE} \\ \mathbf{SOFTWARE} \end{array}$ Mukund Gadgil, Computer Division Version 1, Revision 0 April 28, 1992 #### INTRODUCTION Monitor and Control Module (MCM) is the vital part of the and Monitor System (COMOS) of GMRT. It is a single card data acquisition and control system. MCM cards are distributed at all the remote antennas and at the Central Electronics Building (CEB). All the Monitor and Control points terminate on different MCMs. Each MCM has 64 analog input channels and 16 opto-isolated output lines. The typical monitor points are LO levels, power supply voltages, switch status etc. Typically, each MCM scans the all/specified input analog channels, digitizes the signals and stores the data into the on-chip internal RAM. It can also output the control word at the digital output port. Each MCM is connected to an Antenna Base Computer (ABC) by means of a shared serial link. The serial link between the MCM and the ABC may be either a plastic fiber or a shielded twisted copper pair. Monitor program on MCM receives the data from the ABC via this serial link, decodes it, processes the data and sends the response back to the ABC. This is the basic function cycle of any MCM, the details of which will be made clear further in this report. This report has been divided into four sections as follows: - 1. Overview of MCM Hardware: In this section the hardware of the MCM card is described briefly. Also some details regarding the microcontroller itself are mentioned in order to make the software part understandable. - 2. Overview of Multiprocessor Serial Communication and Communication Protocols: This section narrates the way two microcontrollers (ABC and MCM) communicate with each other and how they exchange the information among themselves. - 3. <u>Overview of MCM Software</u>: This is a brief description of the MCM monitor program. It gives the summary of the MCM operation as far as software is concerned. - 4. <u>Details of the Monitor Program</u>: Detailed description of the MCM Software is given here. It covers all the modules of the monitor program viz. main program, command code routines, interrupt service subroutines etc. # OVERVIEW OF MCM HARDWARE In order to understand the details of the software backbone of MCM, it is necessary to gain a brief knowledge about its hardware. Siemens VLSI Circuit - SAB 80C535 is the heart of the MCM system, which is an 8-bit Microcontroller from the 8031 family with on-chip peripherals viz. timer/counter, 6 X 8-bit ports, 8-bit A/D converter with programmable voltage references, serial interface and 256 bytes of Data Memory. Circuit diagram of the MCM card is as shown in Appendix A. In the current version of MCM hardware, the processor clock frequency is 3.6864 MHz, which results into the 3.26 $\mu \rm sec$ machine cycle time. Input Section: The 64 input lines to MCM are divided into 4 groups of 16 lines by means of four 16-to-1 multiplexers (MUX). MUXes used are Analog Devices's monolithic chips ADG 506. The ADG506 has 16 input lines,out of which one can appear at the output depending on the status of the four control lines of the same. Outputs of the four muxes are in turn connected to four separate analog input lines of the on-chip ADC. <u>ADC Section:</u>80C535 has an on-chip 8-bit A/D converter with 8 multiplexed analog inputs. It operates on the principle of successive approximation by means of capacitive discharge. With a 3.6864 MHz crystal used, the ADC conversion time is approximately $46~\mu sec. (14~processor machine cycles).$ Output Section: 16 digital output lines of each MCM are TTL compatible. However, these outputs are opto-isolated. Desired 16-bit word can be output on these lines for the various control applications viz. electromechanical device control, power switching etc. Internal Data Memory: It can be divided into three physically separate and distinct blocks: the lower 128 bytes of RAM, the upper RAM area(128 bytes) and the 128-byte special function register (SFR) area. While the latter SFR area and the upper RAM area share the same address locations, they must be accessed through different addressing modes as shown in Table I. Table I: Internal Data Memory Map | Address space | Locations | Addressing mode | |------------------------|------------|-----------------| | Lower 128 bytes of RAM | 00H to 7FH | Direct/Indirect | | Upper 128 bytes of RAM | 80H to FFH | Indirect | | Special Function Regs. | 80H to FFH | Direct | From programming point of view, the lower RAM area can be grouped into three address spaces: - 1) A general purpose register area that occupies locations 0 through 1FH. - 2) The next 16 bytes (locations 20H to 2FH) contain 128 addressable bits. - 3) Locations 30H to 7FH may be used as a scratch pad area. Appendix B describes the detailed usage of these 256 byte of internal data memory. Serial Interface: The SAB 80C535 has two serial interfaces which are functionally nearly identical as far as asynchronous communication is concerned. While communicating with the ABC, out of these two channels, serial interface channel-0 is used in half duplex mode. There are four different modes in which the channel can be programmed, out of which Mode 3 (9 bit UART, variable baud rate) mode is programmed by the monitor program of MCM. In this mode 11 bits get transmitted: a start bit(0), 8 data bits(LSB first), a programmable 9th bit, and a stop bit(1). This mode of communication is purposefully selected in order to facilitate the multiprocessor communication in the situation where many MCMs are connected to a single ABC through a shared serial link. Multiprocessor communication will be explained later in the report. <u>Special Function Registers:(SFRs)</u> These registers are located in the internal RAM. These are the registers through which the CPU interfaces with all its peripherals. All control and data transfers from and to peripherals use this register interface exclusively. The monitor program on the MCM has to properly program these registers for a well behaved interface. General Purpose Registers: These are the lower 32 locations of internal RAM. They are grouped into four banks, with each bank consisting of 8 General Purpose Registers (GPRs). Only one of these banks can be enabled at a time. Two bits in the Program Status Word (PSW, is one of the SFRs) PSW.3 and PSW.4, select the active register bank. The 8 GPRs of the selected register bank may be accessed by register addressing in which the instruction opcode indicates which register is to be accessed. For indirect addressing RO and R1 are used as pointers to address internal memory. Table II: GPR Bank Selection | PSW.4 | PSW.3 | Bank selected | |-------|-------|---------------| | 0 | 0 | Bank 0 | | 0 | 1 | Bank 1 | | 1 | 0 | Bank 2 | | 1. | 1 | Bank 3 | #### SECTION II # OVERVIEW OF MULTIPROCESSOR SERIAL COMMUNICATION AND COMMUNICATION PROTOCOLS As it has been already mentioned, a single ABC card communicates with multiple MCM cards via a shared asynchronous serial link. The serial channel between ABC and MCM has to be thus time multiplexed. Each MCM card has got its own address which can be set through a DIP switch at the input port. Monitor program, at on, reads the DIP switch settings (i.e. the address) and into the data memory. Then the mode-3 of the serial selected. If the master ABC wants to transmit a interface-0 is block of data to one of the several salve MCMs, it first sends out an address byte which identifies the target slave. An address byte differs from a data byte in that the programmable 9th bit is set to 1 in address byte and 0 in data byte. With mode-3 selected for MCM card, no interrupt is generated with the data bytes when bit SM20 = 1. (SM20 is one of the eight bits of the SFR SOCON.) An byte, however, will interrupt all slave MCMs, so that address each slave MCM can examine the received byte and see if it is initial part of the SIO handler does the being addressed. The same thing. It compares the address byte sent by the ABC to its own address stored in the internal data memory. If the address match is found, that particular slave MCM clears its SM20 bit and it can get further interrupts with the data bytes. After having received a complete message, the slave sets SM20 again. The addressed leave their SM20 set and go on that were not about their business, ignoring the incoming bytes. Communication Protocols: Communication protocols are nothing but Communication Protocols: Communication protocols are nothing but the fixed formats of the packets that are transmitted by MCM and ABC. In order to extract the information out of these packets this definite packet structure is required. For example, while a packet is sent from ABC to MCM, MCM must know things like packet length, command code, checksum of the packet etc. Accordingly there are different packet filds viz. packet size, id-code, control word etc. which have a definite position and meaning in the packet. Exact protocols for the ABC to MCM and MCM to ABC packets are given in Appendix C. #### SECTION III # OVERVIEW OF MCM SOFTWARE In GMRT COMOS, ABC ultimately controls the operation of MCM in the sense that it can issue certain "commands" to it and get the response back. A detailed description of the commands can be found in the Appendix D. The basic functions of the MCM software are - - i) to accept the command from ABC, - ii) to execute the issued command and - iii) to transmit the response data to ABC. The whole monitor program can be divided into 4 sections: - a] POST and Initializations, - b] Main program, - c] Mode routines and - d] Interrupt Service Routines. These are described in brief below. - a] POST and Initializations: At power on , POST (Power On Self Test) and system initializations are done by the microcontroller. During POST, the CPU checks for the proper functioning of its peripherals. After successful completion of POST, initializations are done during which various SFRs, GPRs, and internal data memory locations are programmed to specific values. - b] Main Program: To start with, function of the main program is to jump into one of the 'mode routines'. After the packet (data bytes) is received completely by the MCM from ABC, program control is transferred to main program again. Main program then performs the following function: - 1) Validates the checksum of the received packet. - 2) Checks whether proper command code (each packet received from ABC contains a command code that corresponds to a specific command to be executed by MCM) and its arguments are received or not. If not, an error flag is raised. - 3) Executes the command. Prepares the core part of the response packet accordingly. - 4) Prepares the header portion of the frame, calculates the checksum, puts it at the tail of the frame and transmits the response frame to ABC. - 5) After transmission is over main programs reinitiates the timer, various data memory locations, SFRs, GPRs and program control is transferred back to the selected mode. - c] <u>Mode Routines</u>: In the first version of the MCM software, there are two modes which the monitor program can exhibit viz. Idle Mode and Scan Mode. In either of the modes a sub-part of the response frame is made ready, serial reception is enabled and then the program enters into the corresponding mode. In idle mode the program does nothing; it just waits for the command to be issued from the ABC. In scan mode, selected analog input channels are scanned, converted into digital form and stored into the RAM. This is a cyclic processes that repeats itself until a packet is received from ABC. - d] <u>Interrupt Service Routines(ISR):</u> There are two types of interrupts on the MCM card: SIO (Serial Input-Output) interrupt and timer interrupt. Accordingly, two ISRs exist. - 1) SIO handler: For the 80C535 microcontroller an interrupt can be generated after each byte is received or transmitted. In the monitor program, both receive as well as transmit interrupts are enabled. There are no separate interrupts for receive/transmit interrupts. It means that the Interrupt Vector Address is the same for both the interrupts. Proper care has to be taken in the SIO handler so as to identify the interrupt source. - 2) Timer handler: Monitor program enables the timer-0 interrupt at power on. Timer-0 is used to detect the receive time-outs. Timer-0 registers (THO and TLO) are loaded with a count value that produces an interrupt after every 5 ms. Timer-0 restarts after every byte is received from ABC and it is stopped after the last byte in the frame is received from ABC. If suppose MCM is expecting a byte from ABC and if it does not get a byte within 5 ms; the timer interrupt gets generated. In the timer handler ISR a flag is set which indicates the mode program to jump into the main program. Main program in turn reports the time-out error to ABC when a next non-timeout ABC-frame is received. #### SECTION IV ### DETAILS OF THE MCM MONITOR PROGRAM What happens at power on ? As is done by all the microcontrollers from the 8031 family, starts executing the program from ROM also 80535 0000H at power on reset. The first instruction has to be a long jump instruction because first fifty ROM locations are reserved interrupt vector addresses. In the first version of the the program this jump is decided to be at location 0030H. So program starts from this location. Initially actual main is checked by writing 1s and 0s into each location internal RAMbyte has been written reading them back to see if correctly or not. If ram test fails, LED on the card is turned port pin p4.0 is set to 1, and serial reception is disabled. After successful completion of ram test, stack pointer is loaded with the value 65H so that now stack area gets defined over the locations 65H to 7FH. After this stack allocation now the program subroutines. The first subroutine is ready to call different called is the timer check routine. Timer Check Subroutine: In this subroutine both the timers, Timer-0 and Timer-1, are checked by putting some count value into the respective count registers and then timers are triggered on. After few 'nop's it is checked if counter values have decremented or not. If not, an error is reported locally by turning on the LED, setting the p4.1 bit and disabling the serial input. After the successful completion of ram and timer checks, a subroutine called init\_80535 is called which initializes the 80535's SFRs. Init 80535: Following sequence of operations is carried out in this subroutine: Select DPTR0 (0th bank of the dptr). Disable both the timers. Enable interrupts. Set the interrupt priorities. (Highest priority to SIO interrupt, second highest priority to Timer-O interrupt and least to ADC interrupt.) Set the serial communication mode to mode-3. Configure timer-1 registers to produce 9600 baud rate. Configure timer-0 to produce 5 ms ticks. Start timer-1. return to the calling program. All these settings are done through different SFRs viz. DPSEL, PCON, IENO, IENO, SOCON etc. After these SFR initializations, GPRs and internal RAM area are also initialized through a subroutine 'init'. <u>Init</u>: The upper and lower ram locations are cleared to all 0s. <u>Thus automatically the entries in the 'mode table' are all set 0s which set the default mode to idle mode</u>. All the flags are also cleared. LED is turned off, MCM address is obtained by reading the dip switch settings and is stored in one of the the ram locations. Register r0 of bank-0 points to the start of the ram area where packet from ABC is to be stored. Then register bank 1 is selected and r0 of this bank points to the 5th location of the ram area which is reserved for MCM to ABC packets. Thus the registers r0 from banks 0 and 1 are the pointers to specific memory locations which get passed to the required parts of the program. Now the main program is ready to jump into one of the modes. Since at power on, idle mode is set, it jumps into idle mode. Idle Mode: Following are the things done in this mode: Prepare the header portion of the response (MCM to ABC) frame. This is done by putting the appropriate numbers at the memory locations pointed to by the memory pointers that in turn are passed by the main program. This header includes the length of logical packet, response code and the arguments. Serial reception is enabled along with the address interrupt. Then the program simply loops until the frame is received from ABC. While in the loop, program continuously checks if 'exit' flag is set or not. If set, it exits out of the loop and jumps into the main program. Scan Mode: Header part of the packet is made ready similar to the idle mode. Program enters into receive mode, enables reception and address interrupt. Then the actual samplings of the channels start. After each sampling is over, it is checked if 'exit' flag is set or not. If set, all the remaining channels are digitized and then the program flow gets diverted back to the main program. If exit flag is not found to be set, sampling is done continuously. Interrupt handling: While the monitor program is in one of the modes, the reception of the data from ABC is handled completely by the SIO isr (interrupt service routine). So the main program does not have to bother about the data reception. Only thing to be done by the main/mode programs is to check for different flags that get set by the isr. Whenever an sio interrupt occurs, it is first seen whether receive interrupt or a transmit interrupt. This can be from the RIO and TIO flags; the flag that is set denotes the type of the interrupt. With a receive interrupt RIO gets set and with a transmit interrupt TIO gets set. For the transmit interrupt, nothing is done except clearing the TIO flag and then program control is transferred to the point where it was before the interrupt. With a receive interrupt, it is checked whether the byte received is the first byte of the packet. If it then that byte is compared with the address of the MCM card. a match is found, then the corresponding flag is set, LED is turned on, and the data interrupt is enabled to receive further bytes. Otherwise it simply returns to the mode program (without enabling the data interrupts.) As soon as the address match is found, timer-0 is triggered to run. If the next data byte does not arrive within the stipulated time period of 5 ms, timer interrupt is generated. In the timer isr, timer-0 is halted, two flags viz. the timeout flag and exit flag are set, and then the program returns to the main/mode program. On the other hand if the next byte is received before the timeout occurs, timer-0 is stopped. Byte is stored, pointer is incremented, and the byte-counter (which is nothing but the no. of bytes received) is incremented. If this count is found to be greater than the maximum count, a flag is set and the pointer is decremented. Further it is checked whether full frame is received or not. If complete packet is received, again a flag is set. If not, isr sets the 'exit' flag whenever only five bytes of the packet are remaining. What happens after the program jumps into main routine from the mode routine ? Program jumps from the mode routine into main program whenever an 'exit' flag is found to be set. After the control is back to the main program, it waits for either frame (from ABC) to be over or timeout to occur. If timeout occurs, following steps are taken: Serial input is disabled. Address interrupt is enabled. Receive mode is enabled. All the GPRs and SFRs are reinitiated. All the flags, except the timeout flag, are cleared. Program control is transferred back to the previous mode. In case the frame is found to be over, first thing that is done is to calculate the checksum. Calculated checksum is compared with the the checksum sent by the ABC. With a mismatch found a flag called 'csumerr' is set. Then the command code is extracted from the packet and it is compared with the existing command codes. If the command code is found to be an unknown one, error is reported by setting the 'icr' (inconsistent command received) flag. Program control is then transferred to that portion of the main program which sends the packet to ABC. With a proper command code found, program jumps into the respective command-code routine. In each such routine again the validity of the arguments is checked. With the invalid arguments found, again the icr flag is set and program control is then transferred to that portion of the main program which sends the packet to ABC. With proper arguments found, that command is executed and it jumps into 'transmit' part of the program. Xmit Program: Before the actual packet transmission starts, the final packet length is calculated, checksum of the whole packet is calculated and is put at the tail of the packet. Control word is also set according to the status of the different flags set previously. Then the complete packet is transmitted in polled fashion. After the transmission is over, program control is transferred back to the mode set recently. #### APPENDIX B ### MEMORY MANAGEMENT ON THE MCM Lower RAM: Range: 00H to 7FH. Used as follows: 00H to 1FH: 32 registers. (4 banks of 8 GPRs) 20H to 22H: 24 bits for different flags. 23H : Used as a 'temp' location in the program. 24H : Version byte 25H : Counter for no. of bytes received from ABC. 26H : Checksum calculated by MCM is stored here. 27H : MCM address is stored at this maloc. 65H to 7FH : Stack locations. Thus in the lower RAM area, memory locations from 28H to 64H (60 locations) are free for use. Upper RAM: Packet received from ABC is stored in this area. It extends from 80H to FFH. 80H to 91H : Frame received from ABC resides in this area. 92H to 99H: Analog mask(8 bytes) stored here. 9AH to 9BH: 16 bit of digital mask. 9CH to 9FH: 32 bits of digital mask. AOH to A7H: 8 bytes for mode table. Packet to be transmitted to ABC starts at location A8H. Maximum packet length (MCM to ABC) can be 5FH(i.e. 95 bytes). So there is no free memory area in upper part. #### APPENDIX C # COMMUNICATION PROTOCOLS - 1) Packet size : 2 bytes, lower byte first. - 2) Identifier Code : 1 byte. - 3) Control Word: 2 bytes, lower byte first. Upper byte does not contain any information. Lower byte is the command code. - 4) Argument Length: 2 bytes, lower byte first. - 5) Actual arguments: These are the variable no. of bytes specified in the argument length. - 6) Checksum :1 byte. It is the last byte of the frame. It is checksum of all the bytes in the frame. Communication from MCM to ABC: Packet structure for MCM to ABC communication is as follows: - 1) Packet size : 2 bytes, lower byte first. - 2) Identifier Code : 1 byte. - 3) Control Word : Lower byte first.Lower byte indicates the communication status Bit 0 : Set when timeout. Bit 1 : Set when checksum error. Bit 2 : Set when an inconsistent command is received. Bit 3 : Set when packet size is too large to fit into the allocated internal data memory buffer. Upper byte indicates the no. of logical packets in the physical packet. The physical packet header contains the above mentioned three fields. So the first logical packet starts at the sixth byte of the physical packet. Logical packet has the following format: - 1) Length of the logical packet: 2 bytes. - 2) Response code: 1 byte. It is the command code to which MCM is responding. - 3) Arguments: These are the variable no. of bytes and are nothing but the arguments sent by the ABC for that particular command. At power on the self test results are available as a single argument byte with following bit mappings: Bit 0 : Timer-0 test. Bit 1 : Timer-1 rest. Bit 2 : ADC test. $Bit(x)=1 \Rightarrow ERROR.$ The last byte of the physical packet will be the checksum byte. #### APPENDIX D ## MCM Executable Commands Basically, there are three types commands that MCM can execute: NULL COMMAND, SET COMMAND and READ COMMAND. The last two commands can have arguments and sub-arguments as explained below. 1] <u>NULL COMMAND:</u> Command code for this command is 00. Whenever this command is received, MCM does not do anything. It simply sends the response to ABC stating its current mode along with the self-test results and remains in that mode only. This command is useful as a handshake command when MCM is not being accessed frequently. - 2] <u>SET COMMAND:</u> Command code = 01. This command has different arguments, sub-arguments and accordingly there are five different SET commands. - a) <u>SET MODE IDLE</u>: Argument -> MODE (00), Sub-argument -> IDLE (00). This command sets the MCM in idle mode. b) <u>SET MODE SCAN</u>: Argument -> MODE (00), Sub-argument -> SCAN (01). This command sets the MCM in scan mode. - c) <u>SET ANALOG MASK</u>: For this command argument is the 'Analog Mask'(01) and sub-arguments are the actual bytes of the mask (8 bytes). - d) <u>Set Digital Mask 16 Bit</u>: Argument -> 'Digital Mask 16 bit' (02), followed by the 2 bytes of the digital mask. - e) <u>Set Digital Mask 32 Bit</u>: Argument -> 'Digital Mask 32 Bit' (03), followed by the 4 bytes of the digital mask. - 3] <u>READ COMMAND</u>: Read command has the command code 02. READ also has five different commands: - a) Read analog mask: Command code -> 02.(02 => read) Argument -> 00 (00 stands for analog mask). - b) Read digital mask 16 bit : Command Code -> 02. Argument -> 01 (01 => digital mask 16 bit). - c) Read digital mask 32 bit : Argument -> 02. - d) Read version : Argument -> 03. Reads the program version. - e) Read mode : Argument -> 04. Reads the current mode in which MCM is looping. Number quoted against an argument is the actual hex code for that particular argument. There are different buffers allocated for analog mask(8 bytes), digital mask 16-bit(2 bytes), digital mask 32-bit(4 bytes), mode(8 bytes) and version(2 bytes). Any SET command copies the respective arguments and sub-arguments into the respective buffers. For SET DIGITAL MASK command MCM outputs the 16/32 bits on the digital output port. Similarly any READ command MCM reads from the respective buffer and sends this data to ABC. Detailed packet formats for all these commands can be found in Appendix. Tree diagram of MCM Commands : # Library of the subroutines/functions used in MCM Software Version 1 1] TMR\_CHK : Timer test subroutine. Input : None. Output : A byte in accumulator. (a) = 00 ==> Timer test passed. = 01 ==> Timer test failed. 2] ADC\_CHK : On-chip ADC test subroutine. Input : None. Output : A byte in accumulator. (a) = 0 ==> ADC test passed. = 1 ==> ADC test failed. 3] INIT\_80535 : Microcontroller specific initializations. Input : None. Output : None. Action: This subroutine when executed does the following things: i) Selects the 0th dptr. ii) Enables and sets the priorities of different interrupts. iii) Configures timers 0 and 1. Timer 0 for generating 5 ms ticks to detect ABC time-outs and timer 1 to generate the 9600 baud. iv) Initializes SOCON SFR in Mode-3. 4] INIT: Software initializations. Input : None. Output : None. Action : i) Clears all the internal RAM locations to 00. (This automatically sets the power-on mode to be idle mode.) ii) Clears all the flags.iii) Reads the MCM address. iv) I nitializes the memory pointers. 5] SET\_CW: Puts control word in the MCM to ABC packet. Input : i) Pointer to the 'control-word' malloc. ii) Software flags. Output : Proper control word set into 'control- word' malloc. - 6] GET\_CHKSUM : Calculate the checksum of the packet. - Input: i) Pointer to the start of the packet. - ii) Packet length including checksum byte. - Output: i) Checksum of the packet. (Appends at the end of packet.) - ii) Packet length in r1(1) - 7] SEND\_BYTE : Sends the byte on the serial link. Input : Byte to be transmitted in accumulator. Output: TIO bit set. There are different segments of the program which are not used as the subroutines. Instead, they are the modules of the program. But these modules can be used as subroutines. Such modules are listed below: - 8] RAM TEST Function: Tests the internal ram of the microcontroller. Alternate 0s and 1s are written into the ram locations and are read back and then checked with the bit pattern written in. If the two match through out the whole ram space, test is a success. Else it is declared to be ram test failure. - 9] **GET\_AD Function:** Configures the ADCONO SFR (i.e. loads the proper byte into it) and DAPR SFR. Starts the conversion cycle of the ADC. SFR bytes can be passed as the i/p parameters and the output parameter is the contents of the ADDAT SFR after conversion is over. - 10] Xmit Function: Input: Start of the packet, Packet length. Output: None. FLOWCHART FOR THE MAIN PROGRAM OF THE HCM MONITOR PROGRAM ``` LISTING OF THE MAIN PROGRAM OF THE MCM SOFTWARE NAME MAIN PROG PROGRAM FOR MONITOR AND CONTROL MODULE program by MUKUND GADGIL ;Initializations done in the file : INIT_MAB.ASM (for 3.6864 MHz crystal) --- * 8/9-bit Format * format flag=0 => 8-bit format ; format flag=1 => 9-bit format table SEGMENT CODE %SET(format_flag,1); /* 9-BIT FORMAT ;THIS IS THE MAIN MODULE ; ----- ;Symbols definitions ;Starting common maloc area 80h pstart in EQU ; for received data init_byte_cnt EQU 00h cntr_init EQU ;Pointer initialisation value. 65h 19 20h 26h 00h ;Starting loc. for stack EQU ____pack_in EQU flag_strt EQU direct_buff EOU EQU stack ;Max pak len.for ABC2MCM frame 0 1 NULL COMMAND EQU SET COMMAND EQU 2 READ COMMAND EQU 3 REBOOT COMMAND EQU 3 EQU max cmd code max set code EQU 4 max_mode_code max_read_code 2 EQU 5 EQU 0 EOU mode code EQU scan code CODE 00H CODE 30h restart start0 CODE sr int sr_int0 CODE 540h timer_0 timer_0_jmp 0bh CODE 600h CODE iadc 43h CODE CODE adc service 700h psize lb DATA pstart in ;packet size lower byte psize hb DATA psize lb+l ;packet size upper byte id code DATA psize hb+l ;ID CODE of the MCM cmd code DATA id code+l ;control word hb DATA cmd code+l arg lenth hb DATA hb+l;Argument length upper byte args ; RAM Storage locations for different packet fields of the frame are :- DATA arg_lenth_lb+1 ; Argument length upper byte DATA arg_lenth_hb+1 ; Actual arguments are stored args ; from this maloc DATA args+l argsl args1 args2 argsl+l DATA ``` DATA canl mask args2+8 ;Analog mask stored here ``` canl mask+8 ;16 bit digital mask stored here DATA dmask 16 dmask_16+2 ;32 bit digital mask stored here dmask_32+4 ;Current Mode byte (1 byte) dmask 32 DATA mode table DATA pstart out DATA mode table+8 ;Start of the Response Packet DATA flag strt flag area temp_loc1 DATA direct buff-2 direct_buff-3 DATA direct_buff-1 DATA version direct_buff ;Counter for no.of bytes byte cntr DATA ;recceived from ABC. direct buff+1 ; Checksum calculated by MCM DATA chk sum ; is stored here. direct_buff+2 ;MCM address stored here. mcm addr DATA ;Flag bit for time out(20.0) 00h tof \mathtt{BIT} csum_err_flag BIT tmr0_status BIT tmr1_status BIT \mathtt{BIT} tof+1 ;Flag set when TimerO NOT ok tof+2 tof+3 ;Flag set when Timer1 NOT ok tmr1_status tof+4 ;Flag set when Timer2 NOT ok tof+5 ;Flag set when internal ram check fails tof+6 ;Set when ADC faulty tof+7 ;Flag which indicates the function to tmr2_status BIT ram status BIT adc status { t BIT} BIT come out ; jump into the main programe. BIT tof+8 ;Set when a too large packet is ptoo large ;received from ABC icr BIT tof+9 ;Flag for inconsistent command. tof+10 ;Flag for selftest. BIT slftst \mathtt{BIT} tof+11 tof+12 ;Flag set when mode is set to idle. BIT ;Actual code sitting in the ROM follows -- RSEG table Command Lookup Table :- cmd_0 ; NULL COMMAND DW lkupt: cmd_1 ;SET COMMAND cmd_2 ;READ COMMAND DW DW cmd 3 ; REBOOT COMMAND DW set_mode set_table: DW set_anl_mask DW set_dmask_2b DW DW set dmask 4b Mode Lookup Table :- idle mode DW modt: scan mode DW rd anlmask read table: DW rd dmask16 DW rd_dmask32 DW DW rd_version DW rd mode ;init 80535 is an external EXTRN CODE (init 80535) :procedure $ INCLUDE (reg535.pdf) $ INCLUDE (rx2.asm) ; Include interrupt handeling ``` \$ INCLUDE (init.pdf) ``` restart CSEG AΤ 1 jmp start ;Main program starts:- start0 CSEG ΑT start: ;Disable reception REN0 clr r0,#1 mov a,#0aah ram loop1: mov @r0,a mov a,@r0 mov cjne a, #0aah, ram_err r0 inc r0, #0ffh, ram loop1 cjne @r0,#0aah mov a, @r0 mov a, #0aah, ram err cjne r0,#1 mov. a,#55h ram loop2: mov @r0,a mov a,@r0 mov cjne a, #55h, ram err r0 inc r0, #0ffh, ram loop2 cjne @r0,#55h mov mov a,@r0 a, #55h, ram err cjne sjmp ram over clr p3.4 ram err: p4,#01 mov ri0 setb jmp sp, #stack ram over: MOV tmr_chk call adc_chk call ;Initialize the 80535 SFRs. init 80535 call main_loop: call init ; After 'init' subroutine is executed REG. BANK 1 is in use. into_mode ljmp psw.5,frame_over into main: jb tof, tot err jb into_main sjmp tot_err: ;Disable reception. clr REN0 %if(%format_flag EQ 1) then ( ;Set SM2 bit to get interrupt s0con.5 setb ; with address byte only. ;Enter into receive mode setb p3.3 ) else ;Enter into receive mode setb p3.3 ) ``` fi ``` clr psw.4 mov r0, #pstart in clr psw.1 clr psw.5 setb psw.3 ; REG. BANK 1 mov r0, #pstart out+5 mov byte cntr, #0 th0, \#0fdh mov mov t10,#0 mov th0, #timer0 hb t10, #timer0 lb mov version, #1 mov mov p5,#2 mov flag_area,#0 mov flag area+1,#0 setb tot 1 jmp into mode frame over: clr REN0 ;Disable reception. setb psw.4 setb psw.3 mov - r1, #80h ;111: r0,#0ffh mov djnz r0,$ r1,111 djnz mov th0,#0 mov t10,#0 setb TR0 ;Start timer-0 clr psw.3 ;Reg. bank 0 clr psw.4 clr p3.4 ; Turn on the LED. After the full frame is received, reg. & memory status are as follows: r0(0) -> points to the malloc immediately following the 'chk_sum' field (byte_cntr) = total no.of bytes in the frame received from \overline{ABC}. dec r0 ;Store the chk sum received ;from ABC into proper field mov chk_sum,@r0 ;i.e. ("chk_sum"). ; Following fragment calculates the actual check sum. ; Calculated check sum will be in the Accumulator -- clr ;Clear the acc. mov rl, byte cntr ;Load r1 with total no. of ; bytes received from ABC. dec r1 ;Decrement r1 by 1 because the ; last i.e.'csum' field in the ;frame has to be ignored while ; calculating the check sum. ded r1 mov r0, #pstart in ;Point to start of frame csum: add a,@r0 inc r0 djnz r1,csum cpl add a,#1h ; Now calculated csum is in acc. cjne a, chk sum, csum err ; Mismatch between the 2 csums. :Report the error. sjmp no csum err ``` ``` csum err: csum err flag setb psw.\overline{3} setb clr psw.4 temp loc, r0 mov r0, #pstart out+3 mov set cw call r0, temp_loc mov r6,#0 mov xmit 1 jmp no csum_err: ; REG. BANK 1 IN USE setb psw.3 psw.4 clr mov temp loc, r0 mov r0, #pstart out+3 call set cw mov r0, temp loc ; REG. BANK 3 IN USE. setb psw.4 r0, #cmd code mov ; Command Code is in the accumulator. mov a,@r0 clr С a, #max_cmd_code subb jnc unknown cmd mov a,@r0 cjne a, #1, not set sjmp cmd 1 unknown cmd: setb icr psw.4 ; REG. BABK 1 clr temp loc, r0 mov r0, #pstart out+3 mov call set cw mov r0, temp_loc r6,#0 mov 1 jmp xmit not set: 1jmp set not cmd 1: setb psw.4 ; REG. BANK 3 ;SET Command found. First check if its mov r1, #args mov a,@r1 ;argument is proper.r1(3) clr С subb a, #max set code ;Unknown argument code found. jnc unknown cmd mov a,@rl ;Proper argument found.Jump to respective rl ;routine. а push acc dptr, #set_table ; Point to the base address of mov ; the command table movc a,@a+dptr mov r1,a ;rl (1) dptr inc acc pop movc a,@a+dptr mov r2,a mov dph,rl dpl,r2 mov clr a @a+dptr ; Jump to the proper SET command routine jmp set mode: ; REG. BANK 1 ;setb psw.3 clr psw.4 ``` ``` ; argument is proper.r1(1) a,@rl mov clr C a, #max mode_code subb ;Unknown argument code found. unknown cmd jnc temp loc, r0 mov r0, #pstart out+3 mov call set cw r0, temp loc mov temp_loc,r0 mov r0, #mode_table mov rl,#argsl mov a,@rl mov @r0,a mov r0, temp loc mov ; Length of 2nd Logical Packet (1.b.) @r0,#5 mov r6,#5 mov inc r0 ; Length of 2nd Logical Packet (h.b.) mov @r0,#0 inc r0 rl, #cmd_code mov a,@rl mov ;Command code (Code:1 -> set)is put @r0,a mov inc r0 mov rl, #args a, @rl mov ;1st argument ( 0 -> mode ) put in lp @r0,a mov r0 inc ;argsl r1 inc a,@rl mov ;2nd argument ( 0/1 -> idle/scan ) @r0,a mov inc r0 1 jmp xmit set anl mask: psw.3 setb psw.4 clr temp loc, r0 mov r0, #pstart_out+3 móv set cw call r0,temp_loc mov temp_loc,r0 mov a,#0 mov r0, #canl mask mov r1, #args1 mov push acc anl cpy: a,@r1 mov @r0,a mov acc pop а inc r0 inc inc r1 cjne a,#8,anl_cpy r0,r2 mov r0, temp_loc mov @r0,#12 ;Length of 2nd Logical Packet (Lower Byte) mov r6,#12 mov inc r0 ;Length of 2nd Logical Packet (Upper Byte) @r0,#0 mov inc r0 rl, #cmd_code mov ``` r1,#argsl mov ;SET MODE found. First check if its ``` mov a, @r1 mov @r0,a ;Command code (Code:1 -> set)is put inc r0 mov r1, #args mov a,#0 arg cpy: push acc mov a, @rl mov @r0,a ;1st argument (1 -> anl mask ) put in lp inc r0 inc r1 ;arqsl pop acc inc а a, #9, arg_cpy cjne r0 inc ljmp xmit set dmask 2b: setb psw.3 clr psw.4 mov temp loc, r0 mov r0, #pstart out+3 call set cw mov r0, temp loc mov r2, r0 mov temp loc,r0 r0,#dmask_16 mov mov r1,#argsl mov a,#0 d16_cpy: push acc mov a,@rl @r0,a mov pop acc inc a inc r0 inc rl cjne a, #2, d16_cpy mov p4,dmask 16 mov p5,dmask, 16+1 mov r0, r2 r0, temp loc mov mov @r0,#6 ;Length of 2nd Logical Packet (Lower Byte) mov r6,#6 r0 inc mov @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) inc r0 mov r1, #cmd_code mov a,@rl mov @r0,a ;Command code (Code:1 -> set) is put inc r0 mov r1, #arqs mov a,#0 arg_cpy1: push acc mov a,@r1 mov @r0,a ;1st argument (1 -> dmask 16 ) put in lp inc r0 inc r1 ;args1 pop acc inc а cjne a,#3,arg_cpy1 inc r0 1 jmp xmit ``` set dmask 4b: ``` setb psw.3 clr psw.4 temp_loc,r0 mov r0, #pstart_out+3 mov set cw call r0, temp loc mov mov r2,r0 temp loc, r0 mov r0, #dmask_32 mov rl,#argsl mov a,#0 mov acc d32_cpy: push a, @rl mov mov @r0,a pop acc inc inc r0 inc rl a, #4, d32 cpy cjne ;Set the 32 bit word to the p4,dmask_32 mov p5, dmask 32+1 mov ; ports. mov p4,dmask 32+2 mov p5.dmask 32+3 r0, r2 mov mov r0, temp loc ;Length of 2nd Logical Packet (Lower Byte) @r0,#8 mov mov r6,#8 inc r0 @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) mov inc r0 r1, #cmd code mov a,@rl mov @r0,a ;Command code (Code:1 -> set)is put mov r0 inc mov r1, #args mov a,#0 arg_cpy2: push acc mov a,@rl ;1st argument (1 -> dmask 16 ) put in lp mov @r0,a inc r0 ;args1 inc r1 pop acc inc cine a, #5, arg_cpy2 r0 inc xmit 1 jmp set not: mov rl, #cmd code mov a, @r1 rl a acc push ;Point to the base address of dptr, #lkupt mov ; the command table a,@a+dptr movc mov rl,a ;r0(0) dptr inc acc pop movc a,@a+dptr r2,a mov dph,rl mov ``` mov dpl,r2 ``` clr ;Jump to the proper routine @a+dptr qmp cmd 0: ; This is the NULL COMMAND. Program just sets the control word, returns the packet length(0,in this case) psw.3 setb psw.4 clr temp loc, r0 mov r0, #pstart out+3 mov setb null call set cw r0, temp_loc mov mov r6,#0 ljmp xmit cmd 2: ;Following fragment of the program executes the command-2 i.e. READ COMMAND ; Read Command found. Check first if ;proper argument is received for this rl,#arcs wow a,@rl ; command. mov clr С a, #max_read_code subb unknown ; Invalid arg.found.Error! jnc setb psw.3 clr psw.4 temp loc, r0 mov r0, #pstart out+3 mov set cw call r0, temp loc mov ; Valid argument found, proceed. r1, #args mov mov a.@rl rl a push acc ;Point to the base address of mov dptr, #read table ; the command table a,@a+dptr movc ;r0(0) mov r1,a dptr inc acc pop a,@a+dptr movc mov r2,a mov dph,rl dpl,r2 mov clr ; Jump to the proper routine @a+dptr jmp unknown: ljmp unknown cmd rd anlmask: setb psw.3 clr psw.4 ;Length of 2nd Logical Packet (Lower Byte) @r0,#12 mov mov r6,#12 inc r0 @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) mov inc r0 ``` ``` r1, #cmd_code mov a,@rl mov @r0,a mov r0 inc r1, #args mov mov a,@r1 mov @r0,a r0 inc a,#0 mov r1, #canl mask mov put msk: push acc mov a,@rl mov @r0,a r0 inc r1 inc acc pop inc а a, #8, put msk cjne ljmp xmit rd dmask16: psw.3 setb psw.4 ; clr ;Length of 2nd Logical Packet (Lower Byte) @r0,#6 mov r6,#6 mov r0 inc ;Length of 2nd Logical Packet (Upper Byte) @r0,#0 mov inc r0 mov r1, #cmd code a,@rl mov @r0,a mov r0 inc rl,#args mov mov a,@rl mov @r0,a r0 inc a,#0 mov rl, #dmask 16 mov push put_d16: acc mov a,@rl mov @r0,a r0 inc r1 inc pop acc inc cjne a,#2,put_d16 1jmp xmit rd dmask32: psw.3 setb clr psw.4 ;Length of 2nd Logical Packet (Lower Byte) @r0,#8 mov r6,#8 mov r0 inc @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) mov inc r0 mov r1, #cmd code a,@r1 mov @r0,a mov r0 inc rl, #args mov a,@rl mov @r0,a mov inc r0 ``` ``` a,#0 mov rl, #dmask 32 mov put d32: push acc mov a,@rl mov @r0,a inc r0 inc r1 acc pop inc а cjne a, #4, put d32 1 jmp xmit rd version: setb psw.3 clr psw.4 ;Length of 2nd Logical Packet (Lower Byte) mov @r0,#5 mov r6,#5 inc r0 mov @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) inc r0 mov r1, #cmd code mov a,@rl @r0,a mov inc r0 mov rl,#args mov a,@rl @r0,a mov inc r0 mov @r0, version inc r0 1jmp xmit rd mode: setb psw.3 clr psw.4 mov @r0,#5 ;Length of 2nd Logical Packet (Lower Byte) mov r6,#5 inc r0 mov @r0,#0 ;Length of 2nd Logical Packet (Upper Byte) inç r0 mov r1, #cmd code mov a,@r1 mov @r0,a inc r0 mov rl,#args mov a,@rl mov @r0,a inc r0 mov r1, #mode_table mov a,@rl mov @r0,a inc r0 l jmp xmit idle mode: ;Prepare the header portion of the response frame :- ; REGESTER BANK 1 IS IN USE. mov @r0,#4 ;llp(lower byte) inc r0 ;Pointer to llp(upper byte) mov r5,#4 mov @r0,#0 ;llp(upper byte) ``` ;Pointer to Response Code inc r0 ``` @r0, #reboot command ; Response code for mov ;Self Test/Reboot ;Pointer to arguments of the response inc r0 @r0,#0 mov mov a,#1 push acc tmr0_status,t0ok jnb orl a,@r0 mov @r0,a t0ok: pop acc rl push acc jnb tmr2 status,t2ok orl a, ero mov @r0,a t2ok: acc pop rl jnb adc status, adc ok a,@r0 orl @r0,a mov adc ok: mov a,@r0 a, #0, stst failed cjne simp stst ok stst failed: p3.4 clr mov p5, #0aah setb ri0 $ dmt stst ok: ;r0(1) points to the Start of the out buff that inc ; will be filled by main prog. if needed byte_cntr, #cntr_init mov clr ri0 ; Enable the RI Interrupt %if(%format flag EQ 1) then ( setb p3.3 ;Enter into receive mode setb s0con.5 ;Set SM2 bit to get interrupt ; with address byte else setb p3.3 ;Enter into receive mode fi setb psw.3 REN0 setb ;Enable reception jnb come out,$ ; Wait till frame received ljmp into main scan mode: ; ======== ; REGESTER BANK USED : 1 ; This command scans those channels for which the corresponding bit in the ; Analog Mask is set to 1. Others are skipped. ; Following are the registers used- ;r2:Counter for no. of mask-bytes covered. ;r3:Global counter for the no. of channel-bits covered ;r4:Counter for the actual no. of channels scanned. ;r7:Counter for no.of bits checked within the mask-byte ; { t r}^0 : { t Points} to the storage locations for ADC values for the different channels ;rl:Points to the analog mask location ``` setb psw.3 ``` clr psw.4 rl, #canl mask mov ;It contains the no.of bytes to be r5,#0 mov ; returned by the mode to the main prog. r2,#8 mov r3,#8 mov a, @rl mov find: rrc а scan jс r3,find scan_ret: djnz r3,#8 mov inc r1 mov a,@rl dinz r2, find sjmp lpl got r5 scan: inc scan_ret sjmp lpl got: p5,r5 mov mov a,r5 add a,#3 a,#8 add add a,#11 a,#13 add r5,a \mathbf{v}om ;Logical Packet length (l.b) mov @r0,a inc r0 @r0,#0 ;Logical Packet length (h.b) mov inc r0 @r0, #set command ; Command code for SET mov inc ; Code for MODE mov @r0, #mode code inc r0 mov @r0, #scan code ;Code for SCAN inc rl, #canl mask mov mov r2,#8 send msk: a,@r1 mov @r0,a mov inc r0 inc r1 djnz r2, send msk %if(%format flag EQ 1) then ( p3.3 ;Enter into receive mode setb ;Set SM2 bit to get interrupt setb s0con.5 ; with address byte else ;Enter into receive mode setb p3.3 fi temp loc1,pl mov ; Make lower nibble of temp loc1 HIGH temp_loc1, #0fh orl clr ADEX ;Internal start of conversion clr ADM ;Stop after 1 conversion. ;Enable reception setb REN0 temp_loc,r0 ; This is the maloc from where the mov ;ADC o/p data will be stored. continue: mov rl, #canl mask ; Counter for no. of mask-bytes covered. r2,#08h mov r3,#00h ;Global Counter for no.of channel-bits ``` mov ``` ; covered. ; Counter for actual no. of channels mov r4,#00h :scanned. r7,#08h ; Counter for no.of bits checked within mov ;a mask-byte. a,@rl n arg: mov clr С rrc wait: ; а get_ad jС rr p4,r3 mov acc.7, get ad jb ; Increment the global counter inc return: come out, chk exit jb djnz not over: r7, wait ; Reload the 'bit-counter' value. mov r7,#8h inc r1 djnz r2,n_arg r0, temp loc mov continue sjmp chk exit: r3, #64, not over ; Wait untill all channels over. cjne cjne r3, #63, not over ; Wait untill all channels over. into main 1 jmp ;Obtaining the ADC o/p for a channel is done below - get ad: p5.7 cpl r4 ;Counter for actual no.of channels scanned inc push acc ;Bit counter value in acc. mov a,r3 b,#16 mov ;After the division acc has div ab ; the mux. no. & b=channel no. a, #0f8h orl p5,a mov ;Set MUX0-2 bits to 1. adcon0,#07h orl ;Proper mux no. has been set adcon0.a anl p4,adcon0 mov mov a,b orl a,#0f0h ; Make upper nibble HIGH a,temp_loc1 anl p1,a mov dapr, #00h mov BSY,$ jb mov a, ADDAT cpl а mov @r0,a inc r0 pop acc ljmp return xmit: setb psw.3 ; REG. BANK 1 clr psw.4 ; Update the packet size value. mov r0, #pstart_out mov a, x5 add a,r6 a,#6 add ``` mov @r0,a ``` ;ppl 1.b. (including csum byte) in r5(1) r5,a mov inc r0 ;ppl h.b. @r0,#0 mov r0 inc mov r1, #id code a,@r1 mov @r0,a mov get chksum call ;Enter into Transmit mode p3.3 clr %if(%format_flag EQ 1) then ( p3.3 clr ;Bit (s0con.3) set for addr. TB80 setb ) else ( ;Enter into transmit mode p3.3 clr ) fi ;Stop timer-0. TR0 clr ; Address byte sent. a, mcm addr mov send byte call delay call :Send Data Bytes now TB80 > clr a,th0 mov send byte call a,t1\overline{0} mov send byte call r0, #pstart out mov a,r2 mov ;ppl in r1(1) rl,a mov a,@r0 sending: mov call send_byte call delay r0 inc rl, sending djnz a,th0 mov send byte call a,t1\overline{0} wow send byte call clr psw.3 psw.4 clr r0, #pstart_in mov clr psw.1 clr psw.5 ;LED Off! p3.4 setb ;Clear all the flags mov flag_area,#0 flag area+1,#0 mov ; REG. BANK 1 psw.3 setb r0, #pstart_out+5 mov byte cntr, #0 mov th0,#0fdh mov t10,#0 mov th0, #timer0_hb mov tl0, #timer0_lb mov version,#1 mov ri0 clr ;Enable recption REN0 setb ti0 clr into mode qmr 1 get_chksum: ; Puts the checksum byte as the last byte of the packet. ;Also returns the packet length in reg. rl(1) ``` ``` r0, #pstart out mov ;This is the packet size without checksum byte dec r5 mov a,r5 rl,a ; (ppl-1) in r1(1) mov inc mov r2.a ;Actual ppl in r2(1) clr а calc: add a,@r0 inc r0 djnz r1, calc cpl inc а ;Checksum in acc mov rl,a ;Checksum stored in rl mov r0, #pstart out mov a,r5 inc ; Packet size including checksum byte а add a,r0 mov r0,a mov a,r1 ;Put the checksum byte at the end of mov @r0,a ;the packet ret send byte: clr ti0 mov sbuf,a ;Get the byte in acc jnb ti0,$ ret init: mov r0, #pstart_in mov r1,#128 do: mov @r0,#00h inc r0 djnz r1,do mov r0, #direct buff r1, #50 mov do1: mov @r0,#0 inc r0 djnz rl,dol psw.3 clr ;Use reg.bank 0 clr psw.4 clr psw.1 clr psw.5 setb p3.4 ; LED Off! mov flag_area,#0 ;Clear all the flags flag area+1,#0 mov mov a,pl anl a,#0f0h r0,#4 mov rot: rr r0,rot djnz mov mcm_addr,a ;Get the MCM Address mov p4, mcm addr mov r0, #pstart_in setb psw.3 ; REG. BANK 1 mov r0, #pstart out+5 mov version,#1 ret set_cw: setb psw.3 ; REG. BANK 1 ``` clr psw.4 ``` mov a,#1 @r0,#0 mov push acc tot, no tof jnb a, @r0. orl mov @r0,a no tof: pop acc rl а push acc csum err flag, no_csum jnb a,@r0 orl mov @r0,a pop no csum: acc rl а push acc jnb icr, no icr orl a,@r0 @r0,a mov no_icr: acc pop rl jnb ptoo large, ok orl a,@r0 @r0,a mov ok: inc r0 null, one_lp jb ; No. of lp.s in pp @r0,#2 mov ret one_lp: @r0,#1 mov ret delay: setb psw.4 r0,#0ffh MOV djnz r0,$ clr psw.4 ret into_mode: rl, #mode_table ;rl(1) mov mov a,@rl rl а push acc dptr, #modt ;Point to the base address of mov ; the mode table a,@a+dptr movc ;r1 (1) mov r1,a inc dptr pop acc a,@a+dptr movc r2,a mov dph,r1 mov mov dpl,r2 clr @a+dptr ; Jump to the proper routine jmp tmr chk: ;This subroutine tests the Timer-0 and Timer-1 ;Testing is done by putting certain values in timer registers and then by ; checking wheather these values decrement or not. clr TR0 ;Disable Timer-0 clr TR1 ;Disable Timer-1 ``` th1, #0ffh mov ;9600 baud ``` tl1, #0ffh mov th0, #0ffh mov t10, #0ffh mov tmod, #gate dable+timer+tmr1_mod2+tmr0_mod1 mov ; Start Timer-\overline{0} setb TR0 r0,#7fh mov djnz r0,$ clr TR0 r0,t10 mov r0,#0ffh,tmr0 ok cjne ;Timer-0 mal-functioning p4,#02h mov clr p3.4 setb ri0 jmр ;Start Timer-1 setb TR1 r0,#7fh mov djnz r0,$ ;Stop Timer-1 clr TR1 mov r0,th0 r0,#0ffh,tmrl ok cine p4,#03h ;Timer-1 mal-functioning mov clr p3.4 ri0 setb jmp ``` tmr1\_ok: tmr0 ok: ret END ``` LISTING OF THE INITIALIZATION PART OF THE MCM SOFTWARE VERSION 1 up 80535 NAME prog 80535 segmer: CODE prog 82535 RSEG DATA(pi,p5,s0cm,pcon,ien0,ien1,ien2,ip0,ip1) EXTRN BIT (BD) EXTRN ; This program does the power on initializations for 80535 ; USED FOR GENERATING $600 BAUD WITH 3.6864 MHz CRYSTAL PUBLIC init 80535 USING init 80535: include(init.pdf) start1: dpsel, # dptr0 mov ;Select dptr0 for ext. mem. ;addressing. TR1 clr ;Disable timerl TR0 clr ;Disable timer0 orl PCON, #8Vh ;Set SMOD bit (PCON.7) IEN0, #int_enable+s_port_int+t0_int_able ;Enable serial IEN1, #1 ;Enable ADC Interrupt mov mov mov ;Enable ADC Interrupt IEN2, #00h mov IPO, #11h ; Highest priority to SIO Int. & second highest: mov ;priority to Tmr0 Int. & lowest to ADC Int. mov %if( %format_flag EQ 1 ) then ( SOCON, #sc_mod3+ser_ip_enable+mul_proc mov SOCON, #sc_mod3+no_ser_ip+mul_proc mov else ( SOCON, #sc_modl+ser_ip_enable+mul_proc mov SOCON, #sc:_mod1+no_ser_ip+mul_proc mov ;Configure Timerl in mode2 (auto reload) for proper baud rate :- clr BD ;Baud rate enable (ADCON0.7) mov th1,#Ofeh ;9600 baud mov th0,#0fdh ;For 5 ms time_out mov t10,#0h th0, #timer0_hb mov mov th0, #timer0 lb tmod, #gat &_dable+timer+tmrl_mod2+tmr0_mod1 mov setb TR1 ;Start timer1 ret ``` ``` LISTING OF THE isr FOR MCM SOFTWARE VERSION 1 ;Serial-int. vector location at {\tt sr\_int} CSEG service ljmp sr/int0 CSEG at service: ri, no ti jb ti0 clr ;return reti ; Following subroutine will be executed if RI interrupt is received ;input:sbuf contains the received byte no ti: psw.1, not 1 jb push acc mov a,sbuf cjne a, mcm addr, end clr p3.4 setb ;Set the flag for address match found psw.1 ; with data bytes. %if(%format flag EQ 1) then ( clr s0con.5 ) else ( clr s0con.5 ;Clear SM2 bit, now interrupt will occur pop acc th0, #0fdh mov mov t10,#0 clr ri0 setb TRO ;Start TMRO only when valid address ;byte is received. reti end: pop acc clr ri0 reti not 1: clr TRO clr psw.3 ;Switch to bank-0 clr psw.4 mov @r0,sbuf ;Store the received byte inc r0 ; Increment the pointer inc byte cntr ; Increment the 'byte-counter' push jb ptoo large, len_nok mov a, byte cntr mov p4, byte cntr cjne a, #max pack in, len ok setb ptoo large len_nok: dec r0 len ok: mov rl, #psize lb mov a,@r1 clr С subb a, byte cntr cjne a, #0, fnot_over setb psw.5 :Frame found to be over, set the flag ``` ``` acc pop psw.3 setb clr tof ri0 clr TR0 clr reti fnot_over: a, #7, return1 cjne come out setb return1: acc pop psw.3 setb tof clr th0, #0fdh wow t10,#0 mov ri0 clr TR0 setb reti timer_0 CSEG ΑT serv_tmr0 1jmp timer_0_jmp CSEG ΑT serv_tmr0: ;Stop Timer-0 clr TR0 p3.4 clr ; Reload the timer values. th0,#0fdh mov t10,#0 mov tof setb come_out setb ``` reti