PIC讀寫93C46程序
;********************************************************************
;*                      PICALC Directives Section                   *
;********************************************************************
 SUBTITL "93C46 3 WIRE INTERFACE ROUTINE"
 LIST     P=16C54,N=40,C=132
;********************************************************************
;*                      Register Assignments                        *
;********************************************************************
indir   equ     0x00            ;Use this register as source/destination for
    ;indirect addressing.
pc      equ     0x02            ;PIC Program Counter.
status  equ     0x03            ;PIC Status Register.
fsr     equ     0x04            ;File Select Register.
serial  equ     0x05            ;Port used for 93C46 control.  Since Port A
    ;is 4 bits wide, we'll use all of Port A.
    ;The following three registers must be
    ;located consecutively in memory.
cmd     equ     0x10            ;This register contains the 2 bit 93C46
    ;command is the upper 2 bit positions and
    ;memory address in the lower 6.
highb   equ     0x11            ;Used in read/write routines to store the
    ;upper byte of a 16 bit 93C46 data word.
lowb    equ     0x12            ;Used in read/write routines to store the
    ;lower byte of a 16 bit 93C46 data word.
cnthi   equ     0x13            ;Used as the upper byte of a sixteen bit loop
    ;counter in RDYCHK routine.
cnt     equ     0x14            ;Used as the lower byte of a sixteen bit loop
    ;counter in RDYCHK routine, and elsewhere as
    ;an eight bit counter.
;********************************************************************
;*                      Bit Assignments                             *
;********************************************************************
carry   equ     0               ;Carry Flag of Status Register.
zflag   equ     2               ;Zero Flag of Status Register.
;For the 3 wire interface, connect the din and dout to the same
;i/o line of the PIC16C5X.
cs      equ     0               ;Port pin tied to CS on 93C46.
din     equ     1               ;Port pin tied to DI on 93C46.
dout    equ     1               ;Port pin tied to DO on 93C46.
clock   equ     2               ;Port pin tied to CLK on 93C46.
;********************************************************************
;*                      General Assignments                         *
;********************************************************************
no_err  equ     0               ;
error   equ     1               ;
tries   equ     0x04            ;After issuing a WRITE, ERASE, ERAL, or WRAL
    ;command, the approximate number of machine
    ;cycles X 256 to wait for the RDY status.
    ;This value must be adjusted for operating
    ;frequencies other than 4 MHz.
read    equ     0x80            ;93C46 Read command.
write   equ     0x40            ;93C46 Write command.
erase   equ     0xC0            ;93C46 Erase command.
ewen    equ     0x30            ;93C46 Erase/Write Enable command.
ewds    equ     0x00            ;93C46 Erase/Write Disable command.
eral    equ     0x20            ;92CXX Erase All command.
wral    equ     0x10            ;92CXX Write All command.
;********************************************************************
;*                      Macro Definitions                           *
;********************************************************************
sel     MACRO                   ;Selects the 93C46 device.
 bsf     serial,cs       ;Chip Select (CS) = '1' to select the device.
 ENDM
dsel    MACRO                   ;De-select the 93C46 device.
 bcf     serial,cs       ;Chip Select (CS) = '0' to de-select the
    ;device.
 ENDM
strtbt  MACRO                   ;Issue the Start Bit to the 93C46.
 bsf     serial,din      ;Start Bit = '1'.
 clkit                   ;Clock it out.
 ENDM
clkit   MACRO                   ;Clocks a serial data bit into or out of the
    ;93C46 device.
 bsf     serial,clock    ;Clock (CLK) = '1'.
 nop                     ;Adjust the number of nop instructions
    ;between the assertion and de-assertion of
    ;CLK in proportion to the PIC operating
    ;frequency.  Refer to the 93C46 data for the
    ;minimum CLK period.
    
 bcf     serial,clock    ;Clock (CLK) = '0'.
 ENDM
;********************************************************************
;*                      Power-On/Reset Entry Point                  *
;********************************************************************
reset_  org     0x1FF
 goto    main
;********************************************************************
;*                      93C46 Routines                              *
;********************************************************************
 org     0x000           ;Locate all subroutines in the lower half of
    ;a Program Memory Page.
;********************************************************************
;*                      DOUT8                                       *
;********************************************************************
    ;Dout8 will output 8 bits of data to the
    ;93C46.  Before calling this routine, the FSR
    ;must point to the byte being transmitted.
dout8   movlw   0x08            ;Initialize loop counter.
 movwf   cnt             ;
d_o_8   bcf     serial,din      ;Assume that the bit to be transfered is a
    ;'0'.  Hence, de-assert DI.
 rlf     indir           ;Rotate the actual bit to be transferred into
    ;the carry bit.
 btfsc   status,carry    ;Test the carry, if our assumption was
    ;correct, skip the next instruction.
 bsf     serial,din      ;No, actual bit was a '1'.  Assert DI.
 clkit                   ;Clock the 93C46.
 decfsz  cnt             ;Repeat until cnt = 0.
 goto    d_o_8           ;Cnt still > 0.
 rlf     indir           ;Restore register to its original condition.
 retlw   no_err          ;Exit with good status.
;********************************************************************
;*                      DIN8                                        *
;********************************************************************
    ;Din8 will input 8 bits of data from the
    ;93C46.  Before calling this routine, the FSR
    ;must point to the register being used to
    ;hold the incomming data.
din8    movlw   0x08            ;Initialize loop counter.
 movwf   cnt             ;
;for the 3 wire interface the direction of the i/o line connected to
;din and dout has to converted from an output to an input.
 movlw   b'00000010'     ;convert RA1 to an input
 tris    serial          ;       /
d_i_8   clkit                   ;Clock a bit out of the 93C46.
 rlf     indir           ;Make room for the incomming bit in the
    ;destination register.
 bcf     indir,0         ;Assume that the incomming bit is a '0' and
    ;clear the LSB of the destination register.
 btfsc   serial,dout     ;Test the incomming bit, if our assumption
    ;was correct, skip the next instruction.
 bsf     indir,0         ;No, actual bit is a '1'.  Set the LSB of the
    ;destination register.
 decfsz  cnt             ;Repeat until cnt = 0.
 goto    d_i_8           ;Cnt still > 0
;for a 3 wire interface, convert the RA1 line back to an output
 movlw   0               ;make RA1 to an output
 tris    serial          ;       /
 retlw   no_err          ;Exit with good status.
;********************************************************************
;*                      RDYCHK                                      *
;********************************************************************
    ;Rdychk will read the 93C46 READY/BUSY status
    ;and wait for RDY status within the alloted
    ;number of processor cycles.  If RDY status
    ;is not present after this set period, the
    ;routine will return with an error status.
rdychk  movlw   tries           ;Initialize time-out counter.
 movwf   cnthi           ;
 clrf    cnt             ;
;for a 3 wire interface, make the RA1 line an input
 movlw   b'00000010'     ;
 tris    serial
 dsel                    ;De-select the 93C46.
;       nop                     ;NOTE:  Check the 93C46 data sheet for
    ;minimum CS low time.  Depending upon
    ;processor frequency, a nop(s) may be
    ;between the assertion and de-assertion of
    ;Chip Select.
 sel                     ;Re-select the 93C46.
notrdy  btfsc   serial,dout     ;If DO is a '0', 93C46 has yet to completed
    ;the last operation (still busy).
 goto    no_error        ;Otherwise RDY status is present within the
    ;alloted time, and return with good status.
 decfsz  cnt             ;No, not yet ready.  Decrement the LSB of our
    ;16 bit timer and check for expiration.
 goto    notrdy          ;Still some time left.  Try again.
 decfsz  cnthi           ;Least significant byte expired - decrement
    ;and check for expiration of the MSB.
 goto    notrdy          ;Still some time left.  Try again.
;for a 3 wire interface, convert RA1 line back to an ouput
 movlw   0               ;convert RA1 to an output
 tris    serial          ;       /
 retlw   error           ;RDY status was not present in the alloted
    ;time, return with error status.
no_error
;for a 3 wire interface, convert RA1 line back to an ouput
 movlw   0               ;convert RA1 to an output
 tris    serial          ;       /
 retlw   no_err
;********************************************************************
;*                      SEE                                         *
;********************************************************************
    ;See will control the entire operation of a
    ;93C46 device.  Prior to calling the routine,
    ;load a valid command/memory address into
    ;location cmd, and for WRITE or WRAL
    ;commands, load registers highb and lowb with
    ;16 bits of write data.  Upon exit, the W
    ;register will contain the completion status.
    ;Only 93C46 instructions which require a
    ;status check can return with an error as the
    ;completion status.  The values that denote
    ;the completion status are defined as
    ;variables 'error' and 'no_err' in the
    ;general assignments section.
see     movlw   cmd             ;Load W with the location of the cmd
    ;register.
 movwf   fsr             ;Transfer that information into the File
    ;Select Register.  The fsr now points to
    ;location cmd.
 sel                     ;Select the 93C46.
 strtbt                  ;Send a start bit.
 call dout8              ;Transmit the 2 bit command and six bit
    ;address.
 btfsc   cmd,6           ;Check for a WRITE or ERASE command.
 goto    see2            ;Yes, parse the command further.
 btfsc   cmd,7           ;Check for a READ command.
 goto    read_           ;Yes, process READ command.
 btfsc   cmd,5           ;Check for a EWEN or ERAL command.
 goto    see3            ;Yes, parse the command further.
 btfsc   cmd,4           ;Check for a WRAL command.
 goto    write_          ;Yes, process WRITE/WRAL command.
exit_   dsel                    ;No further processing required; 93C46
    ;command completed.
 retlw   no_err          ;Return with good completion status.
see2    btfss   cmd,7           ;Check for a ERASE command.
 goto    write_          ;No, process WRITE command.
exit2_  call    rdychk          ;ERASE command requires a status check.
 dsel                    ;De-select the 93C46.
 addwf   pc              ;Compute completion status from results of
    ;status check.
 retlw   no_err          ;Return with good completion status.
 retlw   error           ;Return with bad completion status.
see3    btfsc   cmd,4           ;Check for a EWEN command.
 goto    exit_           ;Yes, no further processing required, exit
    ;now.
 goto    exit2_          ;No, ERAL command which requires a status
    ;check.
read_   incf    fsr             ;Increment the File Select Register to point
    ;to the register receiving the upper byte of
    ;the incomming 93C46 data word.
 call    din8            ;Input the upper byte.
 incf    fsr             ;Increment the File Select Register to point
    ;to the register receiving the lower byte.
 call    din8            ;Input 8 more bits.
 goto    exit_           ;No further processing required, exit now.
write_  incf    fsr             ;Increment the File Select Register to point
    ;to the upper byte of the 16 bit 93C46 data
    ;word to be transmitted.
 call    dout8           ;Output that byte.
 incf    fsr             ;Increment the File Select Register to point
    ;to the lower byte.
 call    dout8           ;Output the lower byte of the 16 bit 93C46
    ;data word.
 goto    exit2_          ;Exit with a status check.
;********************************************************************
;*                      Test Program                                *
;********************************************************************
 main                    ;We've include a sample program to exercise
    ;the PIC to 93C46 interface using a simple
    ;erase, write and varify routine.
 
 clrf    serial          ;Clear the port tied to the 93C46 device.
 movlw   b'11110100'     ;Intialize the data direction register for
 tris    serial          ;that port.
 
 movlw   ewen            ;Load W with the Erase/Write Enable command.
 movwf   cmd             ;Transfer W into cmd register.
 call    see             ;Enable the 93C46 device.
 movlw   eral            ;Load W with the Erase All command.
 movwf   cmd             ;Transfer W into cmd register.
 call    see             ;Erase the 93C46.
 xorlw   error           ;Check completion status.
 btfsc   status, zflag   ;Test for error condition.
 goto    errloop         ;Yes, bad completion status, error-out.
    ;Write loop:
loopcnt equ     0x1F            ;Define an unused location for our test
    ;program loop counter.
tstptrn equ     0xAA            ;Define the test pattern to be written.
 movlw   .64             ;Initialize that counter.
 movwf   loopcnt         ;
 movlw   write           ;Load W with the Write command.
 movwf   cmd             ;Transfer W into cmd register.
 movlw   tstptrn         ;Intialize the 93C46 data registers with
    ;write data.
 movwf   highb           ;
 movwf   lowb            ;
test1   call    see             ;Write data word into 93C46 device.
 xorlw   error           ;Check completion status.
 btfsc   status,zflag    ;Test for error condition.
 goto    errloop         ;Yes, bad completion status, error-out.
 incf    cmd             ;No, increment the 6 bit memory address
    ;field.
 decfsz  loopcnt         ;Have we written all 64 locations?
 goto    test1           ;No, write another location.
;Read loop:
 movlw   .64             ;Initialize loop counter.
 movwf   loopcnt         ;
 movlw   read            ;Load W with the Read command.
 movwf   cmd             ;Transfer W into cmd register.
test2   call    see             ;Read addressed data word from 93C46 device.
 movlw   tstptrn         ;Load W with the pattern written.
 subwf   highb,0         ;Verify the data read against what was
    ;written.
 btfss   status,zflag    ;Same?
 goto    errloop         ;No, error-out.
 movlw   tstptrn         ;Repeat with the lower byte read.
 subwf   lowb,0          ;
 btfss   status,zflag    ;Same?
 goto    errloop         ;No, error-out.
 incf    cmd             ;Yes, both byte correct, increment the 6 bit
    ;memory address field.
 decfsz  loopcnt         ;Have we read all 64 locations?
 goto    test2           ;No, read another location.
allok goto allok ;Home safe!
errloop goto errloop ;Bad news!
 END                     ;Thats all folks!

