aboutsummaryrefslogtreecommitdiffstats
path: root/boards/addons/gdisp/board_SSD1306_spi2.h
blob: 2bc0679f25620bb8ad16530cbc6dfcfa2d90ebf3 (plain)
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
/*
 * This file is subject to the terms of the GFX License. If a copy of
 * the license was not distributed with this file, you can obtain one at:
 *
 *              http://ugfx.org/license.html
 */

/**
 * @file    boards/addons/gdisp/board_SSD1306_spi.h
 * @brief   GDISP Graphic Driver subsystem board interface for the SSD1306 display.
 *
 * @note	This file contains a mix of hardware specific and operating system specific
 *			code. You will need to change it for your CPU and/or operating system.
 */

#ifndef _GDISP_LLD_BOARD_H
#define _GDISP_LLD_BOARD_H

// Pin & SPI setup

#define SPI_DRIVER (&SPID2)
#define SPI_PORT GPIOB
#define SCK_PAD  13
#define MISO_PAD 14
#define MOSI_PAD 15

#define CS_PORT     GPIOC
#define RESET_PORT  GPIOC
#define DNC_PORT    GPIOC
#define CS_PAD     7        // 0 = chip selected
#define RESET_PAD  8        // 0 = reset
#define DNC_PAD    9        // control=0, data=1

static SPIConfig spi_cfg = { NULL, CS_PORT, CS_PAD, 0 };

static GFXINLINE void init_board(GDisplay *g) {
	(void) g;
	g->board = 0; 
	// Maximum speed of SSD1306 is 10Mhz, so set SPI speed less or = to that.  
	//
	// STM32 specific setup
	// STM32_PCLK1 is APB1 frequence in hertz.
	// STM32_PCLK2 is APB2 frequence in hertz.
	// See manual clock diagram to determine APB1 or APB2 for spi in use. 
	// SPI2 uses APB1 clock on stm32151
	// BR bits divide PCLK as follows
	// 000  /2   =   16 MHz
	// 001  /4   =   8 MHz
	// 010  /8   =   4 MHz
	// 011  /16  =   2 MHz
	// 100  /32  =   1 MHz
	// 101  /64  =   500 kHz
	// 110  /128 =   250 kHz
	// 111  /256 =   125 kHz
	unsigned long spi_clk = STM32_PCLK1 / 2;
	unsigned code = 0;
	while (spi_clk > 10000000) {
		code++;
		spi_clk /= 2;
	}
	spi_cfg.cr1 |= (code << 3);
	
	if (g->controllerdisplay == 0) {
		palSetPadMode(SPI_PORT, SCK_PAD, PAL_MODE_ALTERNATE(5)|PAL_STM32_OTYPE_PUSHPULL|PAL_STM32_OSPEED_MID2);
		palSetPadMode(SPI_PORT, MOSI_PAD, PAL_MODE_ALTERNATE(5)|PAL_STM32_OTYPE_PUSHPULL|PAL_STM32_OSPEED_MID2);
		palSetPadMode(SPI_PORT, MISO_PAD, PAL_MODE_ALTERNATE(5));
		palSetPadMode(RESET_PORT, RESET_PAD, PAL_MODE_OUTPUT_PUSHPULL);
		palSetPadMode(CS_PORT, CS_PAD, PAL_MODE_OUTPUT_PUSHPULL);
		palSetPadMode(DNC_PORT, DNC_PAD, PAL_MODE_OUTPUT_PUSHPULL);
		palSetPad(CS_PORT, CS_PAD);
		palSetPad(RESET_PORT, RESET_PAD);
		palClearPad(DNC_PORT, DNC_PAD); 
	}
}

static GFXINLINE void post_init_board(GDisplay *g) {
	(void) g;
}


static GFXINLINE void setpin_reset(GDisplay *g, bool_t state) {
	(void) g;
	palWritePad(RESET_PORT, RESET_PAD, !state);
}

static GFXINLINE void acquire_bus(GDisplay *g) {
	(void) g;
	spiAcquireBus(SPI_DRIVER);
	spiStart(SPI_DRIVER, &spi_cfg); 
	spiSelect(SPI_DRIVER);
}

static GFXINLINE void release_bus(GDisplay *g) {
	(void) g;
	spiUnselect(SPI_DRIVER);
	spiStop(SPI_DRIVER);
	spiReleaseBus(SPI_DRIVER);
}

static GFXINLINE void write_cmd(GDisplay *g, uint8_t cmd) {
	(void) g;
	static uint8_t buf;
	palClearPad(DNC_PORT, DNC_PAD);
	buf = cmd;
	spiSend(SPI_DRIVER, 1, &buf);
}

static GFXINLINE void write_data(GDisplay *g, uint8_t* data, uint16_t length) {
	(void) g;
	palSetPad(DNC_PORT, DNC_PAD);
	spiSend(SPI_DRIVER, length, data);
}


#endif /* _GDISP_LLD_BOARD_H */