**AN85** # Interfacing the X24165/645 to the Motorola 68HC11 Microcontroller by Applications Staff This application note demonstrates how the Xicor X24165/645 family of serial memories can be interfaced to the 68HC11 microcontroller family when connected as shown in Fig. 1. The interface uses two general purpose port D pins to interface to the Serial Memories. Figure 1. Typical hardware connection for interfacing an X24645 to the 68HC11 microcontroller. \*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\* ``` DESCRIPTION: * * ** This file contains general utility routines written in 68HC11 assembly ** language and used to interface the M68HC11 to XICOR Two-wire Serial Memory ** family (X24xxx). The interface between the 68HC11 and X24xxx devices ** consists of a clock (SCL) and a bidirectional data line (SDA). The ** communication interface uses 2 pins from Port D(PD3 = SCL and PD2 = SDA). ** Other components may reside on this bus provided that they do not have the same device identifier byte as the Serial Memory. ** The following table lists all the subroutines in this file with a brief ** description: ** START: Generate the start condition ** STOP: Generate the stop condition * * RESET: Issues the appropriate commands to force device reset ** ProgPage: Transfer from RAM buffer to Serial Memory page ** ProgByte: Transfer the contents of ByteData to the Serial Memory SeqRead: Read multiple bytes, starting from current address pointer RandomRead: Read a byte from a specific memory location * * ACKPoll: Return when the write cycle completes. ** OutACK: Process the acknowledge output cycle ** GetACK: process the acknowledge from the slave device ** The Main program loop programs a test string into the Serial Memory. After ** entire string is programmed, the content of the programmed page is read. ** The read data is stored in the internal RAM. A utility program can be * * written to verify that the buffer content matches the test string. **************************** INTERNAL RAM ******************* EQU $0000 THE INTERNAL RAM BASE ADDRESS (Default) RAMBASE RAM BUFFER ADDRESS RAMBuff EOU RAMBASE RAMBASE+$FF STACK EQU ******************* PROGRAM CONSTANTS ******************* PORT D WOM CONTROL BIT MOWC EOU $20 SDAbit PORT D BITS FUNCTIONING AS BIDIRECTIONAL EQU $04 SCLbit EQU $08 SERIAL DATA (SDA) AND SERIAL CLOCK (SCL) PageNO EQU $00 PAGE NUMBER OF THE Serial Memory BPX EQU $18 BPX BITS POSITION IN WPR WEL EQU $02 WEL BIT POSITION IN WPR $04 RWEL EQU RWEL BIT POSITION IN WPR WPEN EQU $80 WPEN BIT POSITION IN WPR WELon EQU 00000010b WEL CONTROL BYTE EQU 00000110b RWEL CONTROL BYTE RWELon X24165 EQU ``` October, 2000 www.xicor.com 2 of 14 | X24325 | EQU | 0 | | |----------------------------------------|----------------------------------------------|---------------|-------------------------------------------------------------------------------------------| | X24645 | EQU | 1 | NUMBER OF TIMES TO SUPER LOUDING FOR DOLLING | | MaxDelay<br>ByteData | EQU<br>EQU | \$1000<br>58H | NUMBER OF TIMES TO CHECK ACKNOWLEDGE POLLING<br>CHANGES THE x TO AN X IN THE TEST PROGRAM | | SeqReadSize | | 16 | BYTE COUNTS TO SHIFT OUT USING SEQ READ | | seqreausize | EQU | 10 | BITE COUNTS TO SHIFT OUT USING SEQ READ | | IF X241 | 65 | | | | DeviceID | EQU | \$A0 | DEVICE SELECT AND TYPE ID | | HiADDR | EQU | \$0F | MASK FOR UPPER ADDRESS BYTE | | WPR_ADDR | EQU | \$07FF | WPR PHYSICAL ADDRESS LOCATION (BYTE ACCESS) | | PageSize | EQU | 32 | BYTES PER PAGE | | ENDIF | | | | | IF X24325 | | | | | DeviceID | EQU | \$A0 | DEVICE SELECT | | HiADDR | EQU | \$1F | MASK FOR UPPER ADDRESS BYTE | | WPR_ADDR | EQU | \$0FFF | WPR PHYSICAL ADDRESS LOCATION (BYTE ACCESS) | | PageSize | EQU | 32 | BYTES PER PAGE | | ENDIF IF X24645 | | | | | | | | | | DeviceID | EQU | \$80 | DEVICE SELECT | | HiADDRmask | ~ | \$3F | MASK FOR UPPER ADDRESS BYTE | | WPR_ADDR<br>PageSize | EQU<br>EQU | \$1FFF<br>32 | WPR PHYSICAL ADDRESS LOCATION (BYTE ACCESS) BYTES PER PAGE | | 14900120 | 120 | 32 | | | ENDIF | | | | | ************************ | | | | | * INTERNAL REGISTERS AND CONTROL BLOCK | | | | | ***** | ***** | ***** | *************** | | PORTD | EQU | \$08 | PORT D DATA REGISTER | | DDRD | EQU | \$09 | DATA DIRECTION REGISTER FOR PORT D | | SPCR | EQU | \$28 | SPI CONTROL REGISTER | | ***** | ***** | ***** | ************* | | * | | RESET | VECTOR ENTRY POINT | | ****** | ***** | ***** | **************** | | ORG \$FF | \$FFFE RESET VECTOR ADDRESS TO PROGRAM ENTRY | | | | FDB \$E000 JUM | | JUMP TO | BEGINNING OF EXECUTABLE CODE | | * ASSEMBLER | REQUIRE | MENT- CPU TY | PE | | P68H11 | | | | | PAGE | | | | | | | | | | ************************************** | | | | | ************************ | | | | ldy #RAMBuff #### **Application Note** ``` ORG $E000 MAIN: lds * LOAD STACK POINTER #STACK * INITIALIZE THE BUFFER BEFORE PROGRAMMING THE CONTENT TO A PAGE #RAMBuff * IY = RAM BUFFER ADDRESS ldy * IX = TEST STRING ADDRESS ldx #TestString InitRAM: * COPY THE TEST STRING TO ldaa 0,x * RAM BUFFER staa 0,y iny inx tsta bne InitRAM ldx #$1000 * SET REGISTER BASE bset SPCR, X, #DWOM * CONFIG. PORT-D AS OPEN DRAIN * PD2 = SCL ldaa #$0C DDRD,X * PD3 = SDA staa #$FF * CONFIGURE PORT D ldaa staa PORTD, X * RESET THE INTERFACE STATE MACHINE jsr Reset ldd * READ THE WPR CONTENT AND FIND THE #WPR ADDR jsr RandomRead * BLOCKS THAT ARE LOCKED. IF BOTH #WPEN * WPEN BIT AND WP PIN ARE HIGH THEN bita WPEN OFF * BPx BITS ARE PROTECTED (WRITES ARE bne ... WARNING ... * PERMITTED WHEN WP IS BROUGHT LOW). MAKE SURE THAT WP PIN IS LOW BEFORE ATTEMPTING TO WRITE NEW VALUE TO THE WPR WHEN WPEN BIT IS SET. WPEN OFF: * SKIP IF THE BPx BITS ARE bita #BPX * CLEAR (NO BLOCKS ARE PROTECTED) beq NO BPX clra * CLEAR THE BLOCK LOCK BITS (UNPROTECT * THE ENTIRE DEVICE), WAIT FOR jsr ProgBP jsr * WRITE OPERATION TO COMPLETE ACKPoll NO BPX: * SET THE WRITE ENABLE BIT jsr SetWEL ldd #WPR ADDR * READ THE WPR CONTENT AND RandomRead * CHECK THAT WEL BIT #WEL * IS SET HIGH bita bne WRITES EN * ELSE ITS A FAILURE bra * CHECK THE DEVICE/CONNECTIONS*STOP* WRITES EN: ldd #PageNO * D = PAGE NUMBER OF THE Serial Memory ldy #RAMBuff * IY = RAM BUFFER ADDRESS * TRANSFER BUFFER CONTENT TO THE PAGE jsr ProgPage * IN Serial Memory INDICATED BY D(ab) * WAIT TILL COMPLETION OF PAGE PROG. jsr ACKPoll * WRITE BYTE TO SERIAL MEMORY ProgByte jsr ACKPoll * WAIT TILL COMPLETION OF BYTE PROG. jsr ClrWEL * RESET THE WRITE ENABLE BIT jsr #PageNO * D = PAGE NUMBER OF THE Serial Memory ldd ``` \* IY = RAM BUFFER ADDRESS ``` jsr RandomRead * SETUP THE ADDRESS POINTER AND READ 0,Y * FIRST BYTE, SAVE IT TO THE BUFFER staa * ADJUST THE RAM BUFFER POINTER iny #.HIGH.PageNO * LOAD THE UPPER BYTE OF ADDRESS ldaa ldab #$20 * SPECIFY BYTE COUNT FOR SEQ. READ OP * READ/STORE THE REMAINING DATA jsr SeqRead * END OF MAIN bra PAGE ******************* *** Name: SeqRead *** Description: Read sequentially from the Serial Memory *** Function: This subroutine extracts contents of the Serial Memory and stores them into the specified RAM buffer. The total number of bytes to *** read should be provided along with the buffer address. This *** routine assumes that the address pointer has already been *** initialized using the InByte routine. *** Calls: Start, SlavAddr, InByte, OutACK, StopRead *** Input: IY = RAM Buffer Base Address, A = High Order Address *** B = Number of bytes to read *** Output: None *** Register Usage: A, B, IY ************************* SeqRead: jsr Start * START * [C=1] READ OPERATION BIT sec * SEND THE SLAVE ADDRESS BYTE SlavAddr jsr SeqReadNxt: InByte * START READING FROM THE CURRENT ADDRESS jsr staa 0,Y * TOTAL NUMBER OF BYTES TO READ OUT OF inv * Serial Memory decb beq SeqReadEnd * SEND AN ACKNOWLEDGE TO THE DEVICE jsr OutACK SeqReadNxt bra SeqReadEnd: StopRead * END OF READ OPERATION jmp ************************ *** Name: RandomRead *** Description: Reads content of the Serial Memory at a specific location. *** Function: This subroutine sends out the command to read the content of a *** memory location specified in the (D) register. Start, InByte, SlavAddr, OutByte, StopRead *** Calls: *** Input: D = Address of the byte *** Output: A = Read value *** Register Usage: ************************* RandomRead: psha * START jsr Start * [C=0] WRITE OPERATION BIT clc jsr SlavAddr * SEND THE SLAVE ADDRESS BYTE * LOAD THE LOWER BYTE OF THE PAGE tba OutByte * ADDRESS AND SHIFT OUT TO THE DEVICE jsr ``` ``` pula * RECALL HIGH ADDRESS BYTE jsr Start * [C=1] READ OPERATION BIT sec * SEND THE SLAVE ADDRESS BYTE jsr SlavAddr InByte * SHIFT IN A BYTE FROM THE DEVICE jsr qm į * END OPERATION StopRead ************************ *** Name: StopRead *** Description: Terminate read operation *** Function: This subroutine is called at the end of a read operation. The routine generates the last ACK clock cycle followed by a stop *** command. The last ACK bit clock cycle differs from the normal *** ACK bit in that the SDA line is held high. This action notifies the Serial Memory that it should suspend operation. *** Calls: ClockPulse, Stop *** Input: None *** Output: None *** Register Usage: None StopRead: * MAKE SURE THAT THE DATA LINE IS HIGH ... bset PORTD,X,#SDAbit bset DDRD,X,#SDAbit * CHANGE THE PDx DIRECTION TO OUTPUT jsr ClockPulse qmj Stop * END OPERATION PAGE ************************ *** Name: ProgPage *** Description: Update a page of the Serial Memory *** Function: This subroutine transfers the contents of the given buffer to the *** Serial Memory. The caller program must supply the page *** number of the Serial Memory to update and the base address *** of the RAM buffer. *** Calls: Start, SlavAddr, OutByte, Stop *** Input: IY = RAM Buffer Base Address, D(AB) = Page Number *** Output: None *** Register Usage: A,B ******************* ProgPage: jsr Start * START clc * [C=0] WRITE OPERATION BIT jsr SlavAddr * SEND THE SLAVE ADDRESS BYTE * LOAD THE LOWER BYTE OF THE PAGE ADDRESS tba anda #$0E0 * MASK OUT THE UNWANTED LOWER BITS jsr OutByte * AND SHIFT OUT TO THE DEVICE * TRANSFER CONTENT OF THE RAM BUFFER ldab #PageSize ProgPageNxt: * TO THE Serial MEMORY ldaa 0,Y * IY SHOULD BE POINTING TO THE BUFFER jsr OutByte * COVER UP YOUR TRACKS AS BUFFER IS ldaa #$0FF staa 0,Y * READ AND STORED TO THE Serial Memory * TOTAL NUMBER OF BYTES TRANSFERED iny decb * TO THE Serial Memory SHOULD NOT EXCEED ``` ``` bne ProgPageNxt * THE PAGE SIZE jmp * END OF THE OPERATION Stop ******************* *** Name: ProgByte *** Description: Write a byte to serial memory *** Function: This subroutine transfers the contents of ByteData to the Serial Memory. The address written to is conained in the *** slave address and the byte address D(AB). *** Calls: Start, SlavAddr, OutByte, Stop *** Input: D(AB) = Byte Address *** Output: None *** Register Usage: A.B ************************ ProgByte: jsr Start * START clc * [C=0] WRITE OPERATION BIT * SEND THE SLAVE ADDRESS BYTE jsr SlavAddr tba * LOAD THE LOWER BYTE OF THE PAGE ADDRESS #$0E0 * MASK OUT THE UNWANTED LOWER BITS anda * AND SHIFT OUT TO THE DEVICE OutByte jsr * LOAD THE DATA TO BE WRITTEN ldaa #ByteData * SEND OUT DATA TO THE SERIAL MEMORY jsr OutByte jmp Stop ************************ *** Name: EnProqWPR *** Description: Enable updates to Write Protect Register (WPR) *** Function: This subroutine writes the appropriate sequence to the Serial Memory *** to enable updating of the WPR. The ProgWPEN and ProgBP routines * * * must call this subroutine before writes to the WPR are allowed. *** Once this sequence is activated, the only way to exit this mode is by writing to the WPR or resetting the Serial Memory. RandomRead, SetWEL, SetRWEL *** Calls: *** Input: None *** Output: A = INITIAL WPR VALUE *** Register Usage: А, В ************************ EnProgWPR: ldd #WPR ADDR * READ THE WPR CONTENT AND * TEST THE STATUS OF jsr RandomRead bita #WEL * THE WEL BIT AND bne ProgWPR 1 * SKIP IF ITS SET * ALL WRITES TO THE WPR ARE DISALLOWED psha * WHEN THE WEL IS CLEAR, SEND SET WEL jsr SetWEL pula * COMMAND ProgWPR 1: * CHECK THE RWEL BIT AND bita #RWEL * SKIP IF ITS SET bne ProgWPR 2 * WRITING TO BLOCK-LOCK BITS OR WPEN psha * BIT REQUIRE THAT RWEL TO BE SET, jsr SetRWEL * SEND SET RWEL COMMAND pula ProgWPR 2: rts ``` PAGE AddrWPR jsr ``` ******************* *** Name: ProgBP *** Description: Update Block Lock bits in WPR of the Serial Memory *** Function: This subroutine writes to the WPR of the Serial Memory and changes the BP1:0. The caller program must supply the new values *** for the BP1:0 bits. This routine retains the original state of *** the WPEN bit. *** Calls: AddrWPR, EnProgWPR, OutByte, Stop *** Input: A[1:0] = BP[1:0] *** Output: None *** Register Usage: A, IY ************************* ProgBP: anda #$03 * MASK OUT THE UNWANTED BITS * SHIFT THE BPx BITS TO THE asla asla * BIT POSITIONS 4:3 asla * SAVE THE BPx NEW VALUES AND psha * ENABLE WRITING TO THE WPR jsr EnProgWPR #$9A * CREATE THE DATA PATTERN BY MASKING anda #$02 * IN THE DESIRED BIT PATTERN AND oraa * SAVING STATUS OF WPEN BIT tsy * SET THE BPx BITS PER REQUESTED PATTERN 0,y oraa * SAVE THE WPR VALUE ONTO THE STACK staa О,у * GENERATE WPR WRITE COMMAND jsr AddrWPR * SHIFT OUT WPR PATTERN pula jsr OutByte * TO THE DEVICE jmp Stop ******************* *** Name: ProgWPEN *** Description: Update Write Protect Enable bit in WPR of the Serial Memory *** Function: This subroutine writes to the WPR of the Serial Memory and changes the WPEN bit. The caller program must supply the new value of the WPEN bit. The state of the BP1:0 bits are preserved. *** Calls: AddrWPR, EnProgWPR, OutByte, Stop *** Input: *** Output: None *** Register Usage: A, IY ProgWPEN: clra * LOAD THE STATUS FLAGS rora * MASK OUT THE UNWANTED BITS * SAVE THE WPEN BIT NEW VALUE AND psha EnProgWPR * ENABLE WRITING TO THE WPR isr #$9A * CREATE THE DATA PATTERN BY MASKING anda * IN THE DESIRED BIT PATTERN AND #$02 oraa * SAVING STATUS OF WPEN BIT tsy 0,y * SET THE WPEN BIT PER AS REQUESTED oraa * SAVE THE WPR VALUE ONTO THE STACK staa 0,y ``` October, 2000 www.xicor.com 8 of 14 \* GENERATE WPR WRITE COMMAND PAGE ``` * SHIFT OUT WPR PATTERN pula jsr * TO THE DEVICE OutByte jmp Stop ************************ *** Description: Set the Write Enable Latch (WEL) bit in the WPR of the Serial Memory. *** Function: This subroutine writes to the WPR of the Serial Memory and sets the WEL bit. *** Calls: AddrWPR, OutByte, Stop *** Input: NONE NONE *** Output: *** Register Usage: Α ************************* SetWEL: jsr AddrWPR * GENERATE WPR WRITE COMMAND ldaa #WELon * SHIFT OUT WEL-ON PATTERN * TO THE DEVICE jsr OutByte Stop jmp ************************* *** Name: ClrWEL *** Description: Reset the Write Enable Latch (WEL) bit in the WPR of the Serial Memory. *** Function: This subroutine writes to the WPR of the Serial Memory and *** resets the WEL bit. *** Calls: AddrWPR, OutByte, Stop *** Input: NONE *** Output: NONE *** Register Usage: ************************ ClrWEL: * GENERATE WPR WRITE COMMAND jsr AddrWPR * SHIFT OUT WEL-OFF PATTERN clra OutByte * TO THE DEVICE jsr jmp ************************ *** Name: SetRWEL *** Description: Set Register Write Enable Latch bit in the WPR of the Serial Memory. *** Function: This subroutine writes to the WPR of the Serial Memory and *** sets the RWEL bit. *** Calls: AddrWPR, OutByte, Stop *** Input: NONE *** Output: NONE *** Register Usage: A ****************** SetRWEL: AddrWPR * GENERATE WPR WRITE COMMAND jsr #RWELon ldaa * SHIFT OUT RWEL-ON PATTERN * TO THE DEVICE jsr OutByte jmp Stop ``` ``` ************************ *** Name: AddrWPR *** Description: Initiate write operation to the WPR of the Serial Memory. *** Function: This subroutine issues the WPR address and write instruction *** to the Serial Memory. *** Calls: Start, SlavAddr, OutByte *** Input: NONE *** Output: NONE *** Register Usage: A,B ******************* AddrWPR: ldd #WPR ADDR jsr Start * START [ C = OPERATION BIT ] clc * [C=0] WRITE OPERATION BIT SlavAddr * SEND THE SLAVE ADDRESS BYTE jsr tba * LOAD THE LOWER BYTE OF ADDRESS jmp OutByte * AND SHIFT OUT TO THE DEVICE ********************* *** Name: SlavAddr *** Description: Build the slave address for the Serial Memory. *** Function: This subroutine concatenates the bit fields for Device ID, *** the high address bits and the command bit. The resultant *** byte is then transmitted to the Serial Memory. *** Calls: OutByte *** Input: D(AB) = Page number *** C = COMMAND BIT (=0 WRITE, =1 READ) *** Output: None *** Register Usage: ********************* SlavAddr: * MERGE THE COMMAND BIT rola #DeviceID * AND THE DEVICE SELECT BITS eora * WITH THE UPPER BYTE OF anda #HiADDRmask #DeviceID * PAGE ADDRESS eora * SEND THE SLAVE ADDRESS OutByte qmŗ ********************* *** Name: OutByte *** Description: Sends a byte to the Serial Memory *** Function: This subroutine shifts out a byte, MSB first, through the assigned SDA/SCL lines on port D. *** Calls: ClockPulse, GetACK *** Input: A = Byte to be sent *** Return Value: None *** Register Usage: A ************************ OutByte: bset DDRD,X,#SDAbit * CHANGE THE PDx DIRECTION TO OUTPUT sec OutByteNxt: * SHIFT OUT THE BYTE, MSB FIRST rola bcc OutByte0 bset PORTD, X, #SDAbit OutByte1 bra ``` ``` OutByte0: PORTD, X, #SDAbit bclr OutByte1: * CLOCK THE DATA INTO THE Serial Memory jsr ClockPulse #10000000b * MEMORY cmpa * LOOP IF ALL THE BITS HAVE clc bne OutByteNxt * NOT BEEN SHIFTED OUT jmp GetACK * CHECK FOR AN ACK FROM THE DEVICE PAGE ************************* *** Name: InByte *** Description: Shifts in a byte from the Serial Memory *** Function: This subroutine shifts in a byte, MSB first, through the *** assigned SDA/SCL lines on port D. After the byte is received *** this subroutine does not send out an ACK bit to the Serial Memory. ClockPulse *** Calls: *** Input: None *** Return Value: A = Received byte *** Register Usage: A ******************* InByte: ldaa #0000001b * CHANGE THE PDx DIRECTION TO INPUT bclr DDRD,X,#SDAbit InByteNxt: ClockPulse * CLOCK THE Serial Memory AND SHIFT jsr * INTO ACC. THE LOGIC LEVEL ON THE SDA rola bcc InByteNxt * LINE. THE DEVICE OUTPUTS DATA ON SDA, rts * MSB FIRST *** Name: ClockPulse *** Description: Generate a clock pulse *** Function: This subroutine forces a high-low transition on the assigned SCL line on port D. It also samples the SDA *** line state during high clock period. *** Calls: None *** Input: None *** Return Value: C = SDA line status *** Register Usage: None ************************ ClockPulse: bset PORTD, X, #SCLbit * FORCE SCL LINE HIGH. BASED * ON AN 8MHz CRYSTAL FREQ. THE SYSTEM nop nop * BUS CYCLE TIME IS 0.5 MICROSEC. clc brclr PORTD, X, #SDAbit, ClockPulseLo * sec ClockPulseLo: bclr PORTD, X, #SCLbit * LOWER THE CLOCK LINE rts ``` October, 2000 www.xicor.com 11 of 14 ``` *** Name: OutACK *** Description: Send out an ACK bit to the Serial Memory *** Function: This subroutine changes the direction of the SDA pin on port D *** and then clocks an ACK bit to the Serial Memory. The ACK * * * cycle acknowledges a properly received data by lowering the *** SDA line during this period (9th clock cycle of a received *** byte). The direction of the SDA pin is programmed as input prior to returning to the caller. *** Calls: ClockPulse *** Input: None *** Return Value: None *** Register Usage: None ******************** OutACK: bclr PORTD,X,#SDAbit * MAKE SURE THAT THE DATA LINE IS LOW bset DDRD, X, #SDAbit * CHANGE THE PDx DIRECTION TO OUTPUT ClockPulse ******************** *** Name: GetACK *** Description: Clock the Serial Memory for an ACK cycle *** Function: This subroutine changes the direction of the SDA pin on port D *** and then clocks the Serial Memory. It returnes the sampled * * * logic level on the SDA during high clock cycle. The Serial Memory *** acknowledges a properly received command/data by lowering the *** SDA line during this period (9th clock cycle of a transmitted *** byte). If the SDA state is HIGH, it signifies that either it *** did not receive the correct number of clocks or it's stuck in *** previously initiated write command, *** Calls: ClockPulse *** Input: None C = ACKnowledge bit *** Return Value: *** Register Usage: None ******************* GetACK: DDRD,X,#SDAbit * CHANGE THE PDx DIRECTION TO INPUT bclr ClockPulse * CLOCK THE Serial Memory isr DDRD,X,#SDAbit * CHANGE THE PDx DIRECTION TO OUTPUT bset rts PAGE ************************ *** Name: ACKPoll *** Description: Wait for an ACK from the Serial Memory *** Function: This subroutine sends a slave address to the Serial Memory and *** monitors the SDA for an ACK signal. It returns if a low logic level is detected on the SDA during high clock cycle of the acknowledge cycle. The Serial Memory does not respond to any *** commands with an acknowledge bit while the store operation *** is in progress. If no ACK is received another slave address is *** sent to the Serial Memory. The number of iteration is specified * * * by the MaxDelay constant. *** Calls: Start, SlavAddr, Stop *** Input: None ``` ``` C = ACKnowledge bit [=0 ACK ,=1 No ACK was received] *** Return Value: *** Register Usage: A, B, IY ************************ ACKPoll: ldy #MaxDelay * LOAD MAX NO. OF ACK POLLING CYCLE ACKPollnxt: * START THE ACK POLL CYCLE AND Start isr * D = PAGE NUMBER OF THE Serial Memory ldd #PageNO * [C=0] WRITE OPERATION BIT clc * SEND THE SLAVE ADDRESS. THEN jsr SlavAddr * MONITOR THE SDA LINE FOR AN ACK FROM * THE Serial Memory. TERMINATE THE jsr * OPERATION BY A STOP CONDITION. Stop bcc ACKPollExit * EXIT IF THE ACK WAS RECEIVED dey bne ACKPollnxt * LOOP WHILE THE MAXIMUM NO. OF CYCLES * HAVE NOT EXPIRED. ELSE RETURN WITH C=1 ACKPollExit: rts ************************ *** Name: Start *** Description: Send a start command to the Serial Memory *** Function: This subroutine generates a start condition on the bus. The start *** condition is defined as a high-low transition on the SDA * * * line while the SCL is high. The start is used at the beginning *** of all transactions. *** Calls: None *** Input: None *** Return Value: None *** Register Usage: None Start: PORTD, X, #SDAbit * FORCE THE SDA LINE HIGH bset PORTD, X, #SCLbit * FORCE THE SCL CLOCK LINE HIGH bset * BEFORE TAKING THE SDA LOW bclr PORTD, X, #SDAbit nop nop nop nop PORTD, X, #SCLbit * FORCE THE SCL LOW bclr rts *** Name: Stop *** Description: Send stop command to the Serial Memory *** Function: This subroutine generates a stop condition on the bus. The stop *** condition is defined as a low-high transition on the SDA *** line while the SCL is high. The stop is used to indicate end *** of current transaction. *** Calls: None *** Input: None *** Return Value: None *** Register Usage: None ``` END October, 2000 # **Application Note** ``` Stop: * FORCE THE SDA LOW BEFORE TAKING bclr PORTD, X, #SDAbit PORTD, X, #SCLbit * THE SCL CLOCK LINE HIGH bset nop nop nop nop PORTD, X, #SDAbit * FORCE THE SDA HIGH (IDLE STATE) bset rts ******************* *** Name: Reset *** Description: Resets the Serial Memory *** Function: This subroutine is written for the worst case. System interruptions caused by brownout or soft error conditions that reset the main *** CPU may have no effect on the internal Vcc sensor and reset *** circuit of the Serial Memory. These are unpredictable and random events that may leave the Serial Memory interface *** logic in an unknown state. Issuing a Stop command may not be sufficient to reset the Serial Memory. *** Calls: Start, Stop *** Input: None *** Return Value: None *** Register Usage: В ************************* Reset: * APPLY 10 CLOCKS TO THE DEVICE. EACH ldab #$0A ResetNxt: jsr Start * CYCLE CONSISTS OF A START/STOP jsr Stop * THIS WILL TERMINATE PENDING WRITE * COMMAND AND PROVIDES ENOUGH CLOCKS decb * FOR UNSHIFTED BITS OF A READ bne ResetNxt rts * OPERATION TestString: FCC 'xICOR MAKES IT MEMORABLE!' FCB *** END OF X24xxx Serial Memory INTERTERFACE SOURCE CODE ******************* ``` 14 of 14 www.xicor.com