-
Notifications
You must be signed in to change notification settings - Fork 6
/
SPI_master_1.c
158 lines (142 loc) · 5.49 KB
/
SPI_master_1.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
// This is example "SPI_MASTER" code
// for the pic16f877a using the XC8 compiler in MPLAB X
// by Peter Matthews
//
#include <stdio.h>
#include <stdlib.h>
#include <xc.h>
#pragma config FOSC = XT // Oscillator Selection bits (XT oscillator)
#pragma config WDTE = OFF // Watchdog Timer Enable bit (WDT enabled)
#pragma config PWRTE = OFF // Power-up Timer Enable bit (PWRT disabled)
#pragma config BOREN = OFF // Brown-out Reset Enable bit (BOR disabled)
#pragma config LVP = OFF // Low-Voltage (Single-Supply) In-Circuit Serial Programming Enable bit (RB3 is digital I/O, HV on MCLR must be used for programming)
#pragma config CPD = OFF // Data EEPROM Memory Code Protection bit (Data EEPROM code protection off)
#pragma config WRT = OFF // Flash Program Memory Write Enable bits (Write protection off; all program memory may be written to by EECON control)
#pragma config CP = OFF // Flash Program Memory Code Protection bit (Code protection off)
// Set Clock Freq. & Delays
#ifndef _XTAL_FREQ
#define _XTAL_FREQ 4000000
// #define __delay_us(x) _delay((unsigned long)((x)*(_XTAL_FREQ/4000000.0)))
// #define __delay_ms(x) _delay((unsigned long)((x)*(_XTAL_FREQ/4000.0)))
#endif
void main(void);
void pic_init(void);
void spi_init(void);
void pushbutton(void);
void spi_reset(void);
void timer_on(void);
void timer_off(void);
#define BTN01 RB0 // Push Button
#define SCK RC3
#define SDI RC4
#define SDO RC5
unsigned char testCounter = 0;
/****************************************************************************
* main()
****************************************************************************/
void main(){
pic_init();
spi_init();
while(1)
{
__delay_ms(100); // this could be removed to make it quicker!!
pushbutton();
}
}
/****************************************************************************
* pic_init()
****************************************************************************/
void pic_init(){
ADCON1 = 7; // Configure pins as digital
TRISB = 0b00000001; // push button on RB7
nRBPU = 0; // Set weak pullups in port B
/* Timer Interupt Frequency
* f = XTAL_FREQ / 4 * prescalar * Timer ! Resolution
* f = XTAL_FREQ / 4 * 8 * 65536 = 2Hz */
// Timer 1 prescaler 1:8
T1CONbits.T1CKPS1 = 1;
T1CONbits.T1CKPS0 = 1;
}
/****************************************************************************
* spi_init()
****************************************************************************/
void spi_init(void) {
INTCON = 0; // Disable all interrupts
TRISC3 = 0; // SCK=RC3 is the serial clock
TRISC4 = 1; // SDI=RC4 is serial data input
TRISC5 = 0; // SDO=RC5 is serial data output
SSPEN = 0; // Allow Programming of serial port
SMP = 0; // Input data sampled at middle data output time
CKP = 1; // Idle state for clock is a HIGH level
CKE = 1; // Transmit occurs on idle to active clock state
SSPM3 = 0; //
SSPM2 = 0; //
SSPM1 = 0; //
SSPM0 = 0; // SPI Master mode, clock = FOSC/4
SSPIF = 0;
SSPEN = 1; // End programming and Start serial port
}
/****************************************************************************
* pushbutton()
****************************************************************************/
void pushbutton(){
unsigned char dummy;
if(BTN01 == 0){
SSPBUF = testCounter++;
while(BF==0)
{
timer_on();
}
timer_off();
dummy = SSPBUF; //reads SSPBUF
// if(testCounter == 0xff)
// testCounter = 0;
}
}
/****************************************************************************
* isr()
****************************************************************************/
void interrupt isr(void)
{
// Check all interupts - and reset them all !!
if (PIR1bits.TMR1IF == 1) // Check for Timer overflow
{
PIR1bits.TMR1IF = 0; // Reset interupt flag
spi_reset();
timer_off();
}
}
/****************************************************************************
* spi_reset()
****************************************************************************/
void spi_reset(void)
{
unsigned char dummy;
SSPEN = 0; // Reset SPI module
SSPEN = 1; // Reset SPI module
dummy = SSPBUF;
SSPIF = 0;
SSPEN = 0; // Reset SPI module
SSPEN = 1; // Reset SPI module
}
/****************************************************************************
* timer_on()
****************************************************************************/
void timer_on(void)
{
PIE1bits.TMR1IE = 1; // Timer 1 overflow interupt enabled
T1CONbits.TMR1ON = 1; // Timer 1 on
INTCONbits.PEIE = 1; // Enable Peripheral interupts
INTCONbits.GIE = 1; // Global Interupt enable
}
/****************************************************************************
* timer_off()
****************************************************************************/
void timer_off(void)
{
PIR1bits.TMR1IF = 0; // Reset interupt flag
PIE1bits.TMR1IE = 0; // Timer 1 overflow interupt enabled
T1CONbits.TMR1ON = 0; // Timer 1 on
INTCONbits.PEIE = 0; // Enable Peripheral interupts
INTCONbits.GIE = 0; // Global Interupt enable
}