Reliable serial data buffering in 8051

172 Views Asked by At

In the near future, I will be the owner of HM-TRP radio modules that communicate via UART. I will be using an AT89C4051 micro controller as a bridge between the HM-TRP and a bigger circuit that uses an AT89S52 so everything can communicate. The HM-TRP will use a 9600bps baud rate for communication to the AT89C4051 and that chip will use a 22.1184Mhz crystal.

Since I have limited GPIO pins available for the two devices to connect with, I attempted to make a routine that can read a byte from and write a byte to the radio module one nibble at a time. I have designated space for two buffers. One for data to transmit to the module, and one to receive data from the module.

I want to improve this code so that I don't corrupt memory addresses used by other things such as SFRs and stack pointer data. Also the HM-TRP module claims to be half-duplex.

I am trying to make it as asynchronous as possible so that the other processor doesn't have to stall while the bridge (AT89C4051) stalls until the HM-TRP completes sending a byte.

What can I do to improve this? or am I overthinking here?

;uC to radio

DAT equ P1     ;Data channel (only bits 4-7 available)
NIB equ P3.7   ;Nibble Number
RW equ P3.5    ;1=Read, 0=Write
EXEC equ P3.4  ;Execute
R_EN equ P3.3  ;HM-TRP enable
R_CFG equ P3.2 ;HM-TRP config
BUFEND equ 10h ;receive buffer end
BUFENDT equ 3Fh ;transmit buffer end
NIBNO bit 7Fh ;Internal nibble #
IEXEC bit 7Eh ;Internal execution state

;setup everything
setb IEXEC 
setb NIBNO
setb R_EN
setb RS0 ;R0-R7 addresses = high to not corrupt buffers
setb RS1
mov P1,#0FFh ;reset ports
mov P3,#0FFh
mov R0,#BUFEND
mov R1,#BUFENDT
mov R2,#0h  ;R2=remaining transmit characters
mov R3,#0h  ;R3=# received characters
mov SP,#50h ;Get SP out of our space
mov PCON,#80h ;use high-speed UART
mov TH1,#0F4h ;9.6kbps
mov SCON,#50h ;serial mode 1
mov TMOD,#22h ;auto reload for timers
acall delay
clr R_EN      ;Enable HM-TRP radio
setb TR1      ;Start hardware UART
lcall setup

main:
  ;Async routines
  ;Check for received byte and process if any
  jbc RI,recvok
  rtx:
  ;Check for transmit byte and process if any
  jbc TI,xmitok
  txx:
  ;Do user function and repeat
ajmp ufunc

;executed when hardware serial port indicates character transmitted
xmitok:
  mov SBUF,@R1      ;Pass buffered byte to serial
  dec R1        ;adjust pointer
  djnz R2,txx       ;lower count
  mov R1,#BUFENDT   ;when count is 0, reset buffer pointer
  ajmp txx ;back to main routine

;executed when hardware serial port indicates character received
recvok:
  mov @R0,SBUF      ;Store data into buffer right away
  inc R3        ;increment character count
  djnz R0,rtx       ;start buffer at beginning when it reaches end
  mov R0,#BUFEND    
ajmp rtx ;back to main routine

ufunc:
    ;See state of EXEC gpio pin
    jnb EXEC,rexec
      ;pin value needs to be high at first
      clr IEXEC
    rexec:
    jb EXEC,noexec
      jb IEXEC,noexec
    ;only enter if pin was high before set low
      jb RW,d_rd
    ;Here, RW pin = low so Read from HM-TRP is executed
    jb NIB,rnibid
      jb NIBNO,rnibid
        ;NIB pin = low so we want 1st nibble
        mov A,R3
        jz skipread
          ;Only get nibble if at least a byte exists in cache
          mov A,@R0 ;load byte from cache
          inc R0    ;advance pointer
          dec R3    ;lower byte count
          mov R5,A  ;save original byte
          swap A    ;prepare nibble
          anl A,#0F0h ;we want 4 MSB
          mov DAT,A   ;send out 1st nibble
          setb NIBNO  ;prevent hold down of pin from executing this function forever
        skipread:
    rnibid:
    jnb NIB,rnibid2
      jnb NIBNO,rnibid2
        ;We want other nibble
        mov A,R5    ;Get saved byte
        anl A,#0F0h ;But strip lower 4-bits
        mov DAT,A   ;Send out 2nd nibble
        clr NIBNO
    rnibid2:
      d_rd:
      jnb RW,d_wr
    jb NIB,nibid
      jb NIBNO,nibid
        mov A,DAT   ;Read in 1st nibble
        swap A  ;and make it our low nibble
        anl A,#0Fh  ;strip the high part
        mov R6,A    ;and save it
        setb NIBNO  ;prevent hold down of pin from executing this function forever
    nibid:
    jnb NIB,nibid2
      jnb NIBNO,nibid2
        mov A,DAT   ;Read in 2nd nibble
        anl A,#0F0h ;Strip low part
        orl A,R6    ;put both parts together to form byte
        clr NIBNO
        mov @R1,A   ;Store byte in transmit buffer
        dec R1  ;adjust pointer
        inc R2  ;increment number of transmit characters
    nibid2:
      d_wr:
      setb IEXEC ;prevent hold down of EXEC pin from executing things too much
    noexec:
ajmp main

setup:
  ; Setup radio card
  clr R_CFG
  acall delay
  mov R2,#2h ;2 0FFh's = end.
  mov DPTR,#startconfig
  configsu:
    clr A 
    movc A,@A+DPTR ;Read byte from code memory 
    inc DPTR
    inc A 
    jz novalc ;...and don't send it to radio if it's 0FFh
      dec A
      mov R2,#2h ;byte isn't FFh, so restore chances
      clr TI
      mov SBUF,A ;Send byte to radio
      jnb TI,$ ;and stall until its sent
      sjmp configsu
    novalc:
    ;Strike 1+. One more FFh and its over.
    acall delay
  djnz R2,configsu
  ;finish up
  setb R_CFG ;switch radio to data mode
  acall delay
  mov R2,#0h
  clr TI
ret

delay:
  djnz R7,$ ;Delay so radio module can accept setting
ret

;These are codes that get programmed into HM-TRP to get it up and running properly

startconfig:
mhz915:
db 0AAh,0FAh,0D2h,036h,089h,0CAh,0C0h,0FFh
mhz433:
db 0AAh,0FAh,0D2h,019h,0DEh,050h,080h,0FFh
bps9600:
db 0AAh,0FAh,0C3h,000h,000h,025h,080h,0FFh
ubps9600:
db 0AAh,0FAh,01Eh,000h,000h,025h,080h,0FFh
bw105: ;default receive bandwidth: 105khz
db 0AAh,0FAh,0B4h,000h,069h,0FFh
dev35: ;default deviation: 35khz
db 0AAh,0FAh,0A5h,023h,0FFh
xmitdbm: ;xmit power 20dbm max
db 0AAh,0FAh,096h,007h,0FFh
endofconfig:
db 0FFh,0FFh,0FFh,0FFh
0

There are 0 best solutions below