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
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
|
//
// How to access GPIO registers from C-code on the Raspberry-Pi
// Example program
// 15-January-2012
// Dom and Gert
// Revised: 15-Feb-2013
#ifndef __arm__
#define SKIP_GPIO
#endif
// Access from ARM Running Linux
#define BCM2708_PERI_BASE 0x20000000
#define GPIO_BASE (BCM2708_PERI_BASE + 0x200000) /* GPIO controller */
#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <pthread.h>
#define PAGE_SIZE (4*1024)
#define BLOCK_SIZE (4*1024)
#define SER_DAT_PIN 26
#define SER_CLK_PIN 19
#define GATE_PIN 13
#define PAR_CLK_PIN 6
#define SENSE_PIN 5
int mem_fd;
void *gpio_map;
typedef struct {
char buf[2][35];
int should_buf, is_buf, keep_running;
} t_display_data;
// I/O access
volatile unsigned *gpio;
// GPIO setup macros. Always use INP_GPIO(x) before using OUT_GPIO(x) or SET_GPIO_ALT(x,y)
#define INP_GPIO(g) *(gpio+((g)/10)) &= ~(7<<(((g)%10)*3))
#define OUT_GPIO(g) *(gpio+((g)/10)) |= (1<<(((g)%10)*3))
#define SET_GPIO_ALT(g,a) *(gpio+(((g)/10))) |= (((a)<=3?(a)+4:(a)==4?3:2)<<(((g)%10)*3))
#define GPIO_SET *(gpio+7) // sets bits which are 1 ignores bits which are 0
#define GPIO_CLR *(gpio+10) // clears bits which are 1 ignores bits which are 0
#define GPIO_ALTER(a) *(gpio+(a==0?10:7)) // alters (a==0: clear, else: set) bits which are 1 ignores bits which are 0
#define GET_GPIO(g) (*(gpio+13)&(1<<g)) // 0 if LOW, (1<<g) if HIGH
#define GPIO_PULL *(gpio+37) // Pull up/pull down
#define GPIO_PULLCLK0 *(gpio+38) // Pull up/pull down clock
void setup_io();
void drop_privileges();
void *put_on_display(void *param);
void printButton(int g)
{
if (GET_GPIO(g)) // !=0 <-> bit is 1 <- port is HIGH=3.3V
printf("Button pressed!\n");
else // port is LOW=0V
printf("Button released!\n");
}
int main(int argc, char **argv)
{
pthread_t thread_id;
t_display_data display_data = {
"abcdefghijklmnopqrstuvwxyzabcdefghi",
"ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHI",
0,
0,
1
};
#ifndef SKIP_GPIO
// Set up gpi pointer for direct register access
setup_io();
#endif
// Drop root privileges
drop_privileges();
/************************************************************************\
* You are about to change the GPIO settings of your computer. *
* Mess this up and it will stop working! *
* It might be a good idea to 'sync' before running this program *
* so at least you still have your code changes written to the SD-card! *
\************************************************************************/
#ifndef SKIP_GPIO
// must use INP_GPIO before we can use OUT_GPIO
INP_GPIO(SER_DAT_PIN);
OUT_GPIO(SER_DAT_PIN);
INP_GPIO(SER_CLK_PIN);
OUT_GPIO(SER_CLK_PIN);
INP_GPIO(GATE_PIN);
OUT_GPIO(GATE_PIN);
INP_GPIO(PAR_CLK_PIN);
OUT_GPIO(PAR_CLK_PIN);
INP_GPIO(SENSE_PIN);
#endif
pthread_create(&thread_id, NULL, put_on_display, &display_data);
for (int i=0; i<20; i++) {
usleep(500000);
if (display_data.should_buf == display_data.is_buf) {
for (int j=0; j<35; j++)
display_data.buf[1-display_data.is_buf][j] -= 1;
}
display_data.should_buf ^= 0x01;
}
display_data.keep_running = 0;
pthread_join(thread_id, NULL);
return 0;
} // main
//
// Set up a memory regions to access GPIO
//
void setup_io()
{
/* open /dev/mem */
if ((mem_fd = open("/dev/mem", O_RDWR|O_SYNC) ) < 0) {
perror("can't open /dev/mem");
exit(-1);
}
/* mmap GPIO */
gpio_map = mmap(
NULL, //Any adddress in our space will do
BLOCK_SIZE, //Map length
PROT_READ|PROT_WRITE,// Enable reading & writting to mapped memory
MAP_SHARED, //Shared with other processes
mem_fd, //File to map
GPIO_BASE //Offset to GPIO peripheral
);
close(mem_fd); //No need to keep mem_fd open after mmap
if (gpio_map == MAP_FAILED) {
perror("mmap error");
exit(-1);
}
// Always use volatile pointer!
gpio = (volatile unsigned *)gpio_map;
} // setup_io
//
// Drop (root) privileges
//
void drop_privileges()
{
/* Drop superuser privileges in correct order */
if (setgid(99) == -1) {
perror("can't drop group privileges");
exit(-1);
}
if (setuid(99) == -1) {
perror("can't drop user privileges");
exit(-1);
}
} // drop_root
//
// Shift the bits provided in content into the display
//
void *put_on_display(void *param)
{
t_display_data *display_data = param;
char *raw_output;
int line;
raw_output = malloc(7*(5+1));
if (raw_output == NULL) {
perror("malloc failed");
exit(-1);
}
while (display_data -> keep_running) {
#ifdef SKIP_GPIO
printf("\n");
#endif
display_data -> is_buf = display_data -> should_buf;
for (line=0; line<7; line++) {
raw_output[line*6] = 1<<line;
memmove(raw_output+1+6*line,&(display_data -> buf[display_data -> is_buf][5*line]),5);
}
for (line=0; line<7; line++) {
#ifndef SKIP_GPIO
GPIO_CLR = 1<<GATE_PIN; // Licht an
#endif
for (int byte=0; byte<6; byte++) {
#ifndef SKIP_GPIO
for (int bit=0; bit<8; bit++) {
GPIO_CLR = 1<<SER_CLK_PIN;
GPIO_ALTER((*(raw_output+6*line+byte)>>bit) & 0x01) = 1<<SER_DAT_PIN;
GPIO_SET = 1<<SER_CLK_PIN;
}
#else
if (byte==0)
printf(" %02x ", *(raw_output+6*line+byte));
else
printf("%c", *(raw_output+6*line+byte));
#endif
}
usleep(1000);
#ifndef SKIP_GPIO
GPIO_SET = 1<<GATE_PIN; // Licht aus
GPIO_CLR = 1<<PAR_CLK_PIN;
GPIO_SET = 1<<PAR_CLK_PIN;
#endif
}
}
free(raw_output);
return NULL;
} // put_on_screen
|