Bonjour à tous,
J'ai un petit pb dans mon projet. Avant tout je tiens à dire que je suis débutant en programmation (ça fait 1 mois que j'y touche) donc soyez indulgent svp. Tout ce qui suit, a été fait en partie grâce au cours de Bigonoff (je n'ai pas tout lu non plus) que je remercie...
Je traduit en mot ce que j'ai programmé :
J'ai demandé à mon PIC de faire 8 conversions analogique/digitale et d'en faire la moyenne. Cette moyenne je veux l'afficher sur des 7 segments. Pour l'afficher j'ai programmé un TIMER qui toutes les 20 ms renvoie à l'adresse 0x08 de l'interruption. L'interruption consiste donc à afficher la valeur tens et ones sur les 7 segments. Donc toutes les 20 ms les cadrans des dixaines et des unités s'allumeront pour afficher les valeurs stockées dans ones et tens. Seulement quand je fais "watch" dans MPLAB, je ne vois pas le PORTB ni le PORTA changer... Les deux restent à 0 et je comprends pas pourquoi...
Pouvez vous m'aider?
Merci
Voilà mon code:
LIST p=18F2525
#include <p18F2525.inc>
CONFIG OSC = INTIO67
CONFIG FCMEN = OFF
CONFIG IESO = OFF
CONFIG PWRT = OFF
CONFIG BOREN = OFF
CONFIG WDT = OFF
CONFIG MCLRE = OFF
CONFIG LPT1OSC = OFF
CONFIG PBADEN = OFF
CONFIG DEBUG = OFF
;***************************** ******************** ********************
; Masques *
; *
;***************************** ******************** ********************
MASK_ADCON0 EQU B'00000000' ; la conversion analogique n'est pas autorisée
MASK_ADCON1 EQU B'00001110' ; Le voltage de référence pour la conversion étant comme VSS et VDD
MASK_ADCON2 EQU B'10001100' ; définie la justification par la droite ou gauche
MASK_TRISA EQU B'00000011' ; les pins 2 et 3 sont configurées comme des entrées
; et 4,5,6,7,9 et 10 comme des sorties
MASK_TRISB EQU B'00000000' ; Toutes les pins PORTB sont configurées comme des sorties
MASK_TRISC EQU B'01000000' ; Toutes les pins PORTC sont configurées en sortie sauf RC6 qui sera utilisé
; pour l'USART
MASK_T0CONOFF EQU B'00000111' ; le timer n'est pas enclenché
MASK_T0CONON EQU B'10000111' ; le timer est enclenché
MASK_INTCONOFF EQU B'00000000' ; interdire les interruptions
MASK_INTCONON EQU B'10100000' ; autorise uniquement l'interruption du timer 0
MASK_TIMER0ON EQU B'10000111' ; lancer le timer
;***************************** ******************** ********************
; DEFINITION *
;***************************** ******************** ********************
#DEFINE bouton PORTA,2 ; Bouton permettant de mettre sous tension
#DEFINE a PORTB,0 ; Afficheur de la barre a
#DEFINE b PORTB,1 ; Afficheur de la barre b
#DEFINE c PORTB,2 ; Afficheur de la barre c
#DEFINE d PORTB,3 ; Afficheur de la barre d
#DEFINE e PORTB,4 ; Afficheur de la barre e
#DEFINE f PORTB,5 ; Afficheur de la barre f
#DEFINE g PORTB,6 ; Afficheur de la barre g
#DEFINE dp PORTB,7 ; Afficheur de la barre dp
#DEFINE Ones PORTC,0 ; Allume le 7 segments des unités
#DEFINE Tens PORTC,1 ; allume le 7 segments des dixaines
;***************************** ******************** ********************
; TABLE 7 segments *
;Cette table permet le stockage d'informations concernant l'affichage*
; des 7 segments. Elle contient les adresses qui permettent d'affiher*
;tel ou tel chiffre. Les adresses suivantes sont attribuées *
;respectivement à 0,1,2,3,4,5,6,7,8,9,10 *
;***************************** ******************** ********************
tabl_segment EQU 0x0001000
org tabl_segment
db 0x3F,0x06,0x5B,0X4F
db 0x26,0x6D,0x7D,0x07
db 0x7F,0x6F,0x00
;***************************** ******************** ********************
; VARIABLES *
;***************************** ******************** ********************
CBLOCK 0x000
cmptac : 1 ; Temps nécessaire à la charge du convertisseur pour l'
; acquisition
cmptac50 : 1
cmptmoy : 1 ; compteur permettant d'avoir 8 échantillons et d'en faire
; la moyenne
accumL : 1
accumH : 1
moyenneL : 1 ; 8 premiers octets de la moyenne
moyenneH : 1 ; 8 derniers octets de la moyenne
ones : 1 ; variables des unités pour 7-segments
tens : 1 ; variables des dixaines pour 7-segments
casel : 1 ; permet de sélectionner quel cadran allumer
ENDC
;***************************** ******************** ********************
; INTERRUPTION *
;***************************** ******************** ********************
; AFFICHAGE ;
; Cette interruption sert à afficher toutes ;
; les 20ms sur les cadrans 7 segments les ;
; données stockées dans les registres ones ;
; et tens ;
;***************************** *****************
org 0x08
AFFICHAGE
movlw UPPER(tabl_segment) ; charger bits 16 à 21 de l’adresse
movwf TBLPTRU ; dans pointeur UPPER
movlw HIGH(tabl_segment) ; charger bits 8 à 15 de l’adresse
movwf TBLPTRH ; dans pointeur HIGH
movlw LOW(tabl_segment) ; charger bits 0 à 7 de l’adresse
movwf TBLPTRL ; dans pointeur LOW
BCF PORTC,0 ; On configure RC0 et RC1 en sortie
BCF PORTC,1
movlw 0x01 ; on met 1 dans w
movwf casel ; on charge 1 dans casel
cpfseq casel,0 ; comparer casel à w, et sauter s'ils sont égaux
bra AFITENS ; si différent alors on va afficher les dixaines
movf ones,w,0 ; mettre le nombre d'unité dans w
addwf TBLPTRL,1,0 ; ajouter ones dans le pointeur de la table L
tblrd * ; lire la table à l'adresse du pointeur
movf TABLAT,w ; mettre le résultat de la table dans w
movwf PORTB ; envoyer la valeur de la table vers l'afficheur 7 segments
BSF PORTC,0,0 ; allumer uniquement le cadran des unités
BRA NVCASEL ; aller à NVCASEL de façon à sélectionner un nouveau cadran
AFITENS
movlw 0x02 ; charger 2 dans w
cpfseq casel,0 ; comparer si casel est égal à deux
; si oui alors sauter l'instruction suivante
bra NVCASEL ; sinon aller à NVCASEL
movf tens,w,0 ; charger les tens dans w
addwf TBLPTRL,1,0 ; Ajouter les tens dans le pointeur de la table
tblrd * ; lire la table
movf TABLAT,w ; charger la valeur de la table pointer dans w
movwf PORTB ; envoyer la valeur vers l'afficheur
BSF PORTC,1,0 ; allumer uniquement le cadran des dixaines
NVCASEL
RLNCF casel,1,0 ; faire une rotation de bit vers la gauche
movlw 0x02 ; si casel > 2 alors le remettre à 1
cpfsgt casel,0 ; si = à 2 alors retourner pour afficher les dixaines
bra AFITENS
movlw 0x63 ; charger w pour le timer low
movwf TMR0L ; charger le registre low du timer
movlw 0xFF ; charger w pour le timer high
movwf TMR0H ; charger le registre high du timer
movlw MASK_INTCONON ; Autoriser les interruptions
movwf INTCON,0
;***************************** ******************** ********************
; INITIALISATION *
;***************************** ******************** ********************
init
movlw MASK_ADCON1 ; ANO = entrée analogique. Vref = Vss et Vdd
movwf ADCON1
movlw MASK_ADCON0 ; AN0 sélectionnée, pas de conversion en cours
movwf ADCON0
movlw MASK_ADCON2 ; Justification par la droite, Acquisition time = 2Tad
movwf ADCON2
movlw MASK_TRISA ; Configurer Pin 2 et 3 comme des entrées
movwf TRISA
clrf PORTB
movlw MASK_TRISB ; Configurer toutes les pins PORTB comme des sorties
movwf TRISB
clrf PORTC
movlw MASK_TRISC ; configurer les pins 11,12,13,14,15,16,18 en sortie et 17 en entrée
movwf TRISC
movlw MASK_TIMER0ON ; Lancer le timer0
movwf T0CON,0
bra MAIN
;ATTENTION JE NE COMPRENDS PAS CE QU'A FAIT A-S
;***************************** ******************** ********************
; SOUS ROUTINE *
;***************************** ******************** ********************
; calculer la moyenne
;***************************** ******************** *********************
MOYENNE
RRCF accumH ; divise RRNCF et RRCF par 2
RRNCF accumL
RRCF accumH ; Divise encore par 2
RRNCF accumL
RRCF accumH ; divise encore par 2 donc finalement on a par 8
RRNCF accumL
bra CHIFFRE
;***************************** ******************** ********************
; CHIFFRES *
;***************************** ******************** ********************
CHIFFRE
clrf ones ; Effacer "ones" et "tens"
clrf tens
DIXAINE
movlw 0xA ; Charger 10 dans w pour savoir
; combien de dixaine il y a dans notre moyenne
subwf accumL,1,0 ; pour se faire, on soustrait 10 à accumL et on laisse
; le résultat dans accumL jusqu'à obtenir un négatif
movlw 0x0
subwfb accumH,1,0 ; sousraire le Report de soustraction si il y en a
bn NEGATIF ; si le résultat < 0 alors N du reg status vaut 1
; Donc on va direct à NEGATIF. Si > 0
incf tens ; alors on incrémente de 1 la variable tens
BZ FIN_CHIFFRE ; si le résultat = 0 alors plus de reste donc terminé
bra DIXAINE ; Si le résultat est possitif alors on recommence.
NEGATIF
movlw 0xA ; on rajoute les 10 dans accumL pour retrouver un chiffre positif
addwf accumL,1,0 ; et les passer dans la variable ones
UNITE
movff accumL,ones
FIN_CHIFFRE
bra MAIN
;***************************** ******************** ********************
; PROGRAMME PRINCIPALE *
;***************************** ******************** ********************
MAIN
movlw 0x63 ; l'interruption aura lieu toutes les 20msec
movwf TMR0L
movlw 0xFF
movwf TMR0H
;movlw MASK_INTCONON ; autoriser les interruptions du timer
; movwf INTCON,0
CONVERSION
clrf accumL
clrf accumH ; réinitialiser accum à 0
movlw MASK_INTCONOFF ; Interdire les interruptions
movwf INTCON,0
bsf ADCON0,ADON ; autoriser la conversion analogique
movlw 0xFA ; charger le registre W à 250
movwf cmptac ; charger la variable cmpt1 pour obtenir un échantillon toutes les 25ms
movlw 0x8 ; charger le w à 8 pour obtenir 8 échantillons
movwf cmptmoy ; charger cmptmoy à 8
BOUCLE
; toutes les 50ms prendre un échantillon
movlw 0x02 ; charger 2 dans w
movwf cmptac50
BOUCLE50MS
decfsz cmptac ; décrémenter cmptac
bra BOUCLE50MS ; =/ 0 boucle1
bsf ADCON0,GO_DONE ; lancer la conversion analogique/digitale
decfsz cmptac50
bra BOUCLE50MS
BOUCLECONV
btfsc ADCON0,GO/DONE ; tester si = 0 alors conversion finie
; une fois la conversion finie, le résultat
; vient se mettre dans deux registres : ADRESL et ADRESH
bra BOUCLECONV ; =/0 donc boucle2
movlw 0x01 ; simuler une acquisition
movwf ADRESL
movlw 0x00 ; simuler une acquisition
movwf ADRESH
; STOCKER LES ECHANTILLONS POUR FAIRE LA MOYENNE
;***************************** ******************** ********************
movf ADRESL,w ; charger la conversion ADRESL dans registre w
addwf accumL ; additionner la précédente conversion avec la valeur de accumL
movf ADRESH,w ; charger la conversion ADRESH dans registre w
addwfc accumH ; additionner la précédente conversion avec la valeur de accumH
; retour à la prise d'échantillons
;***************************** ******************** ********************
BOUCLEMOY
decfsz cmptmoy ; décrémenter le registre cmptmoy de 1
bra BOUCLE ; si =/ 0 alors retour à BOUCLE 1
bra MOYENNE
END
-----