;=============================================
;Синтезатор 145 мгц
;А.Ю. Денисов <alldn@yahoo.com> Тамбов 1999 г.
;ПЧ=10700.    WDT OFF.	 F_RC_osc=1.2 мГц.
;
;=============================================
;порты
;
indf	equ	00  ;указатель косвенной адресации
tmr0	equ	01  ;таймер
pcl	equ	02  ;счетчик команд
stat	equ	03  ;регистр состояния status
fsr	equ	04  ;регистр косвенной адресации
;
port_a	equ	05  ;порт А
port_b	equ	06  ;порт В
;
;=============================================
;регистры
;
step	equ	0E  ;шаг перестройки
count	equ	0F  ;счетчик
count1	equ	10  ;счетчик
rotate	equ	11  ;ротация DATA
cross	equ	12  ;признак ретранслятора
;
H_mulplr  equ	13  ;Старший байт 16-битного множимого
L_mulplr  equ	14  ;Младший байт 16-битного множимого
mulcnd	 equ	15  ;8-битный множитель
;
R0	equ	16  ;
R1	equ	17  ;на
R2	equ	18  ;индикаторе
;
H_byte	equ	19  ; буфер
M_byte	equ	1A  ; преобразования
L_byte	equ	1B  ;
;
rxhi	equ	1C  ;делитель
rxlo	equ	1D  ;приема
;
txhi	equ	1E  ;делитель
txlo	equ	1F  ;передачи
;
;=============================================
;Макросы
;
jnz	macro	metka
	btfss	stat,2
	goto	metka
	endm
;
jz	macro	metka
	btfsc	stat,2
	goto	metka
	endm
;
jnc	macro	metka
	btfss	stat,0
	goto	metka
	endm
;
jc	macro	metka
	btfsc	stat,0
	goto	metka
	endm
;
;=============================================
	list p=16F84
       __CONFIG 03FF3H
;=============================================
;************ SUBROUTINES ********************
;=============================================
;
	org	0
	goto	start
;
;Начальная инициализация
init
	movlw	10	 ;задержка
	movwf	count1	 ;для
init2			 ;инициализации
	movlw	0xFF	 ;индикатора
	movwf	count	 ;HT1610
init1			 ;
	decfsz	count,f  ;
	goto	init1	 ;
			 ;
	decfsz	count1,f ;
	goto	init2	 ;
init3
	movlw	0fh	 ;
	movwf	fsr	 ;
clear			 ;начальное
	clrf	indf	 ;обнуление
	incf	fsr,f	 ;служебных
	btfss	fsr,5	 ;ячеек
	goto	clear	 ;
;
	clrf	fsr	 ;выбор 0 банка
;
	movlw	0x71	 ;
	movwf	txhi	 ;передача
	movlw	0x84	 ;145.300/5
	movwf	txlo	 ;
;
	movlw	0x69	 ;
	movwf	rxhi	 ;прием
	movlw	0x28	 ;(145.300-10.7)/5
	movwf	rxlo	 ;
;
	retlw	0	 ;выход из п\программы.
;
;=============================================
;Выдать байт на индикатор
displ
	movwf	rotate	  ;байт из W в файл rotate
	clrf	count	  ;загрузить 8
	bsf	count,3   ;	в счетчик бит.
	bcf	stat,0	  ;сброс cary
displ1
	clrf	stat
	rlf	rotate,f  ;сдвиг rotate влево через C
	btfsc	stat,0	     ;если Cary=0 skip
	bsf	port_b,5  ;установить DATA=1
	btfss	stat,0	     ;если Сary=1 skip
	bcf	port_b,5  ;уст. DATA=0
	nop
	bsf	port_b,4  ;CLK=1 _-
	nop
	bcf	port_b,4  ;   =0   -_ тактовый импульс
	decfsz	count,f   ;проверить счетчик бит.
	goto	displ1	  ;не равен 0. Следующий бит.
;
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Вкл/откл режима ретранслятора
retrans
	movf	cross,f ;ретранслятор включен?
	jnz	init3	;да, нужно выключить
	movlw	0xE0	;иначе нужно
	movwf	cross	;включить
;
	movlw	0x71  ;
	movwf	txhi  ;передача
	movlw	0x6B  ;145.175/5
	movwf	txlo  ;
;
	movlw	0x69  ;
	movwf	rxhi  ;прием
	movlw	0x87  ;(145.775-10.7)/5
	movwf	rxlo  ;
;
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Плюс 25 килогерц
plus25
	movf	step,w
	addwf	rxlo,f
	jnc	pl1
	incf	rxhi,f
pl1
	movf	step,w
	addwf	txlo,f
	jnc	pl2
	incf	txhi,f
pl2
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Минус 25 килогерц
minus25
	movf	step,w
	subwf	rxlo,f
	jc	min1
	decf	rxhi,f
min1
	movf	step,w
	subwf	txlo,f
	jc	min2
	decf	txhi,f
min2
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Плюс 100 килогерц
plus100
	movf	step,w
	movwf	count
	bcf	stat,0
	rlf	count,f
	rlf	count,w
	addwf	rxlo,f
	jnc	pl21
	incf	rxhi,f
pl21
	movf	step,w
	movwf	count
	bcf	stat,0
	rlf	count,f
	rlf	count,w
	addwf	txlo,f
	jnc	pl22
	incf	txhi,f
pl22
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Минус 100 килогерц
minus100
	movf	step,w
	movwf	count
	bcf	stat,0
	rlf	count,f
	rlf	count,w
	subwf	rxlo,f
	jc	min21
	decf	rxhi,f
min21
	movf	step,w
	movwf	count
	bcf	stat,0
	rlf	count,f
	rlf	count,w
	subwf	txlo,f
	jc	min22
	decf	txhi,f
min22
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Подтверждение нажатия клавиши
sound
	movlw	.50
	movwf	count
sound2
	bsf	port_b,6  ;_i-
;
;задержка --------------------------------
	movlw	.10
	movwf	count1
sound1
	decfsz	count1,1    ;задержка ~2ms
	goto	sound1
;
	bcf	port_b,6  ; -i_
;
;задержка --------------------------------
	movlw	.10
	movwf	count1
sound11
	decfsz	count1,1    ;задержка ~2ms
	goto	sound11
;
	decfsz	count,1
	goto	sound2
;
	retlw	0	  ;выход из п\программы.
;=============================================
;Установить частоту в 1508PL1
setfreq
	movlw	4	   ;выдать в 1508ПЛ1
	call	send	   ;1й байт=4 (делитель на 2000)
	movfw	count1
	movwf	fsr
	movfw	0
	call	send	   ;
	incf	count1,f
	movfw	count1
	movwf	fsr
	movfw	0
	call	send	   ;
strob
	bsf	port_a,2   ;1 _-
	nop
	bcf	port_a,2   ;0	-_
;
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Выдать байт из rotate последовательно в 1508PL1
send
	movwf	rotate	  ;байт из W в файл rotate
	clrf	count	  ;загрузить 8
	bsf	count,3   ;	в счетчик бит.
	bcf	3,0	  ;сброс cary
next
	rlf	rotate,f  ;сдвиг rotate влево через C
	btfsc	3,0	  ;если Cary=0 skip
	bsf	port_a,0  ;установить DATA=1
	btfss	3,0	  ;если Сary=1 skip
	bcf	port_a,0  ;уст. DATA=0
	nop
	bsf	port_a,1  ;CLK=1 _-
	nop
	bcf	port_a,1  ;   =0   -_ тактовый импульс
	decfsz	count,1   ;проверить счетчик бит.
	goto	next	  ;не равен 0. Следующий бит.
	retlw	0	  ;выход из п\программы.
;
;=============================================
;Преобразование bin to BCD
;
b2_bcd
	bcf	stat,0	  ; clear the carry bit
	movlw	.24
	movwf	count
	clrf	R0
	clrf	R1
	clrf	R2
loop16	rlf	L_byte,f
	rlf	M_byte,f
	rlf	H_byte,f
	rlf	R2,f
	rlf	R1,f
	rlf	R0,f
;
	decfsz	count,f
	goto	adjDEC
;
; заменяем нули на 0xA для LCD индикатора
	movlw	0x0F
	andwf	R2,0
	jnz	next1
	movlw	0x0A
	iorwf	R2,f
next1
	movlw	0xF0
	andwf	R2,0
	jnz	next2
	movlw	0xA0
	iorwf	R2,f
next2
	movlw	0x0F
	andwf	R1,0
	jnz	next3
	movlw	0x0A
	iorwf	R1,f
next3
	retlw	0
;
adjDEC
	movlw	R2
	movwf	fsr
	call	adjBCD
;
	movlw	R1
	movwf	fsr
	call	adjBCD
;
	movlw	R0
	movwf	fsr
	call	adjBCD
;
	goto	loop16
;
adjBCD	movlw	3
	addwf	0,W
	movwf	count1
	btfsc	count1,3  ; test if result > 7
	movwf	0
	movlw	30
	addwf	0,W
	movwf	count1
	btfsc	count1,7  ; test if result > 7
	movwf	0	  ; save as MSD
;
	retlw	0
;
;=============================================
; Умножение 16-битного числа на 8-битное
;
m16x8
	clrf	H_byte
	clrf	M_byte
	clrf	L_byte
	movlw	.16
	movwf	count
	movf	mulcnd,w
	bcf	stat,0	  ; Clear the carry bit in the
loop			  ; status Reg.
	rrf	H_mulplr,f
	rrf	L_mulplr,f
	btfsc	stat,0
	addwf	H_byte,f
	rrf	H_byte,f
	rrf	M_byte,f
	rrf	L_byte,f
	decfsz	count,f
	goto	loop
;
	retlw	0
;
;=============================================
; Изменение шага перестройки 5 <-> 25 килогерц
;
newstep
	movlw	4
	xorwf	step,f
	goto	hotstart
;
;=============================================
;************ MAIN PROGRAM *******************
;=============================================
start
	bsf	stat,5	 ;для 16F84
	movlw	0
	movwf	01h
	movlw	8
	movwf	port_a
	movlw	08Fh
	movwf	port_b
	bcf	stat,5

	call	init	   ;инициализация
	movlw	5
	movwf	step
;
	goto	hotstart
;
;опрос клавиш
begin
	movf	port_b,w   ;из port_b в W
	andlw	8f	   ;маскируем выходные порты
	xorlw	0f	   ;есть нажатая клавиша?
	btfsc	stat,2	   ;
	goto	begin

	movlw	10	   ;
	movwf	count1	   ;
loop2			   ;
	movlw	0xFF	   ;
	movwf	count	   ;
loop1			   ;
	decfsz	count,f    ;задержка (антидребезг)
	goto	loop1	   ;
			   ;
	decfsz	count1,f   ;
	goto	loop2	   ;
;
;	     еще раз смотрим нажатие клавиши
	movf	port_b,w   ;из port_b в W
	andlw	8f	   ;маскируем выходные порты
	xorlw	0f	   ;есть нажатая клавиша?
	btfsc	stat,2	   ;
	goto	begin
;
; обработка нажатия клавиши ----------------------
	movf	port_b,w   ;читаем из port_b в W
	movwf	rotate	   ;запоминаем в rotate
	andlw	0x03
	jz	retrans1

	movf	rotate,w
	andlw	0x0C
	jz	newstep

	rlf	rotate,w
	jc	transmit

	rrf	rotate,f
	jnc	pl25

	rrf	rotate,f
	jnc	mn25

	rrf	rotate,f
	jnc	pl100

	call	minus100   ;минус 100 кгц
hotstart
	call	sound	   ;писк
	movlw	rxhi	   ;установка относительной
	movwf	count1	   ;адресации
	call	setfreq    ;установка частоты
; прибавим ПЧ для вывода на индикатор
;
	movlw	0x5C	   ;загружаем первое слагаемое
	movwf	L_mulplr   ;равное 10700/5
	movlw	0x08
	movwf	H_mulplr
;
	movf	rxlo,w	   ;второе слагаемое равное
	addwf	L_mulplr,f ;(частота - ПЧ)/5
	jnc	hot1
	incf	H_mulplr,f
hot1
	movf	rxhi,w
	addwf	H_mulplr,f
hot2
	movlw	.5	       ;умножаем на 25
	movwf	mulcnd
	call	m16x8
	call b2_bcd   ;переводим в дв.-десятичный вид
;	с исправлениями для индикатора HT1610
;
;			 и выводим на индикатор
	movf	cross,w ;признак ретранслятора
	call	displ	;
	movlw	0	; пробел
	call	displ	;
	movf	R0,w	; частота
	call	displ	;
	movf	R1,w	;
	call	displ	;
	movf	R2,w	;
	call	displ	;
;
keyoff			   ;ждем отжатие клавиши
	movf	port_b,w   ;из port_b в W
	andlw	8f	   ;маскируем выходные порты
	xorlw	0f	   ;есть нажатая клавиша?
	jnz	keyoff
;
	movlw	10
	movf	count1,f
koff2
	movlw	0xFF
	movf	count,f
koff1
	decfsz	count,f
	goto	koff1

	decfsz	count1,f
	goto	koff2

	goto	begin
;
retrans1		   ;ретранслятор on/off
	call	retrans
	goto	hotstart
;
pl25			   ;плюс 25 кгц
	call	plus25
	goto	hotstart
;
mn25			   ;минус 25 кгц
	call	minus25
	goto	hotstart
;
pl100			   ;плюс 100 кгц
	call	plus100
	goto	hotstart
;
transmit
	movlw	txhi	   ;установка относительной
	movwf	count1	   ;адресации
	call	setfreq    ;установка частоты
;
	movf	txlo,w	   ;переносим множимое
	movwf	L_mulplr
	movf	txhi,w
	movwf	H_mulplr
;
	movlw	.5	       ;умножаем на 25
	movwf	mulcnd
	call	m16x8
	call b2_bcd  ;переводим в дв.-десятичный вид
;	с исправлениями для индикатора HT1610
;	и выводим на индикатор
	movf	cross,w ;признак ретранслятора
	call	displ
;
	movlw	0	; пробел
	call	displ
;
	movf	R0,w	; частота
	call	displ
	movf	R1,w
	call	displ
	movf	R2,w
	call	displ
;
keyoff1     ;ждем отжатие клавиши
	movf	port_b,w   ;из port_b в W
	andlw	8f	   ;маскируем выходные порты
	xorlw	0f	   ;есть нажатая клавиша?
	btfss	stat,2	   ;
	goto	keyoff1

	movlw	10
	movf	count1,f
koff12
	movlw	0xFF
	movf	count,f
koff11
	decfsz	count,f
	goto	koff11

	decfsz	count1,f
	goto	koff12
;
	goto	hotstart
	end
Hosted by uCoz