-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathrtems-brm.c
421 lines (342 loc) · 10.5 KB
/
rtems-brm.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
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
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
/* Simple BRM interface test.
* Program can be compiled in one of three modes:
* ¤ BC - Bus controller
* ¤ RT - Remote Terminal
* ¤ BM - Bus monitor
*
* Gaisler Research 2007,
* Daniel Hellström
*
*/
#include <rtems.h>
#define CONFIGURE_INIT
#include <bsp.h> /* for device driver prototypes */
rtems_task Init( rtems_task_argument argument); /* forward declaration needed */
/* configuration information */
#define CONFIGURE_APPLICATION_NEEDS_CONSOLE_DRIVER
#define CONFIGURE_APPLICATION_NEEDS_CLOCK_DRIVER
#define CONFIGURE_APPLICATION_NEEDS_NULL_DRIVER 1
#define CONFIGURE_MAXIMUM_TASKS 8
#define CONFIGURE_RTEMS_INIT_TASKS_TABLE
#define CONFIGURE_EXTRA_TASK_STACKS (3 * RTEMS_MINIMUM_STACK_SIZE)
#define CONFIGURE_LIBIO_MAXIMUM_FILE_DESCRIPTORS 16
#define CONFIGURE_INIT_TASK_PRIORITY 100
#include <rtems/confdefs.h>
/* Configure Driver manager */
#if defined(RTEMS_DRVMGR_STARTUP) && defined(LEON3)
/* Add Timer and UART Driver for this example */
#ifdef CONFIGURE_APPLICATION_NEEDS_CLOCK_DRIVER
#define CONFIGURE_DRIVER_AMBAPP_GAISLER_GPTIMER
#endif
#ifdef CONFIGURE_APPLICATION_NEEDS_CONSOLE_DRIVER
#define CONFIGURE_DRIVER_AMBAPP_GAISLER_APBUART
#endif
#endif
#define CONFIGURE_DRIVER_AMBAPP_GAISLER_PCIF /* PCI is for GR-RASTA-IO and GR-701 B1553BRM */
#define CONFIGURE_DRIVER_AMBAPP_GAISLER_GRPCI /* PCI is for GR-RASTA-IO and GR-701 B1553BRM */
#define CONFIGURE_DRIVER_PCI_GR_RASTA_IO /* GR-RASTA-IO PCI TARGET has a B1553BRM core */
#define CONFIGURE_DRIVER_PCI_GR_701 /* GR-701 PCI TARGET has a B1553BRM core */
#define CONFIGURE_DRIVER_AMBAPP_MCTRL /* Driver for Memory controller needed when using SRAM on PCI board */
#ifdef LEON2
/* PCI support for AT697 */
#define CONFIGURE_DRIVER_LEON2_AT697PCI
/* AMBA PnP Support for GRLIB-LEON2 */
#define CONFIGURE_DRIVER_LEON2_AMBAPP
#endif
#define CONFIGURE_DRIVER_AMBAPP_GAISLER_B1553BRM /* B1553BRM Driver */
#define RASTAIO_B1553RBM_REMOTE_ADDR
#define RASTA_IO_SRAM
#include <drvmgr/drvmgr_confdefs.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
/* Include driver configurations and system initialization */
#include "config.c"
#include <b1553brm.h>
#include "brm_lib.h"
/* Select BRM core to be used in sample application.
* - /dev/b1553brm0 (First ON-CHIP core)
* - /dev/b1553brm1 (Second ON-CHIP core)
* - /dev/rastaio0/b1553brm0 (The BRM core on first GR-RASTA-IO board)
* - /dev/rastaio1/b1553brm0 (The BRM core on second GR-RASTA-IO board)
* - /dev/gr701_0/b1553brm0 (The BRM core on first GR-701 board)
* - /dev/gr701_1/b1553brm0 (The BRM core on second GR-701 board)
*/
#define B1553BRM_DEVICE_NAME "/dev/b1553brm0"
/*#define B1553BRM_DEVICE_NAME "/dev/rastaio0/b1553brm0"*/
//#define BRM_BC_TEST
rtems_task task1(rtems_task_argument argument);
rtems_task task2(rtems_task_argument argument);
/*extern int errno;*/
rtems_id Task_id[3]; /* array of task ids */
rtems_name Task_name[3]; /* array of task names */
/* =========================================================
initialisation */
rtems_task Init(
rtems_task_argument ignored
)
{
rtems_status_code status;
system_init();
printf("******** Starting Gaisler BRM test ********\n");
Task_name[1] = rtems_build_name( 'T', 'S', 'K', 'A' );
Task_name[2] = rtems_build_name( 'T', 'S', 'K', 'B' );
status = rtems_task_create(
Task_name[1], 1, RTEMS_MINIMUM_STACK_SIZE * 2,
RTEMS_DEFAULT_MODES,
RTEMS_DEFAULT_ATTRIBUTES, &Task_id[1]
);
status = rtems_task_create(
Task_name[2], 1, RTEMS_MINIMUM_STACK_SIZE * 2,
RTEMS_DEFAULT_MODES,
RTEMS_DEFAULT_ATTRIBUTES, &Task_id[2]
);
/* Starting receiver first */
status = rtems_task_start(Task_id[1], task1, 1);
status = rtems_task_delete(RTEMS_SELF);
}
#ifdef BRM_BC_TEST
/* Bus controller */
/* =========================================================
task1 BC */
/* Execution list length */
#define MSG_CNT 4
int proccess_list(brm_t chan, struct bc_msg *list, int test){
int j,ret;
if ( brmlib_bc_dolist(chan, list) ){
printf("LIST EXECUTION INIT FAILED\n");
return -1;
}
/* Blocks until done */
printf("Waiting until list processed\n");
if ( (ret=brmlib_bc_dolist_wait(chan)) < 0 ){
printf("LIST EXECUTION DONE FAILED\n");
return -1;
}
if ( ret != 1 ) {
/* not done */
printf("LIST NOT PROCESSED\n");
return -1;
}
/* done */
printf("List processed.\n");
for (j = 1; j <= MSG_CNT; j++) {
if (list[j-1].ctrl & BC_BAME)
printf("Error msg %d, %x: BAME\n", j, list[j-1].ctrl);
}
return 0;
}
rtems_task task1(
rtems_task_argument unused
)
{
brm_t chan;
struct bc_msg cmd_list[MSG_CNT+1];
struct bc_msg result_list[MSG_CNT+1];
int j,k;
printf("Starting task 1: BC mode\n");
/* open device */
chan=brmlib_open(B1553BRM_DEVICE_NAME);
if ( !chan ){
printf("Failed to open device driver 0\n");
rtems_task_delete(RTEMS_SELF);
return;
}
/* before starting set up
* ¤ blocking mode
* ¤ BC mode
*/
/* Set BC mode */
printf("Task1: Setting BC mode\n");
brmlib_set_mode(chan,BRM_MODE_BC);
/* total blocking mode */
printf("Task1: Setting TX/RX blocking mode\n");
brmlib_set_block(chan,1,1);
printf("Setting up command list.\n");
/* Begin execution list loop */
while (1) {
/* Set up messages to RT receive subaddresses */
for (j = 1; j <= MSG_CNT; j++) {
cmd_list[j-1].rtaddr[0] = 1;
cmd_list[j-1].subaddr[0] = j;
cmd_list[j-1].wc = 8;
cmd_list[j-1].ctrl = BC_BUSA; /* RT receive on bus a */
for (k = 0; k < 9; k++){
cmd_list[j-1].data[k] = 0;
}
/* message input */
cmd_list[j-1].data[1] = 'G';
cmd_list[j-1].data[2] = 'R';
cmd_list[j-1].data[3] = (j-1)+7;
}
cmd_list[MSG_CNT-1].wc++;
cmd_list[MSG_CNT].ctrl |= BC_EOL; /* end of list */
/* Set up RT transmit sub addresses (request RTs to send answer) */
for (j = 1; j <= MSG_CNT; j++) {
result_list[j-1].rtaddr[0] = 1;
result_list[j-1].subaddr[0] = j;
result_list[j-1].wc = 8;
result_list[j-1].ctrl = BC_BUSA | BC_TR; /* RT transmit on bus a */
/* clear data */
for (k = 0; k < 9; k++){
result_list[j-1].data[k] = 0;
}
}
result_list[MSG_CNT-1].wc++;
result_list[MSG_CNT].ctrl |= BC_EOL; /* end of list */
printf("------------- BC: START LIST EXECUTION -------------\n");
printf("Start CMD list processing.\n");
if ( proccess_list(chan,cmd_list,0) ){
sleep(1);
continue;
}
printf("Sleeping 20s\n");
sleep(20);
printf("------------- BC: START LIST EXECUTION -------------\n");
printf("Start RESULT list processing.\n");
/* Clear data that input will overwrite */
for (j = 1; j <= MSG_CNT; j++) {
/* clear data */
for (k = 0; k < 8; k++){
result_list[j-1].data[k] = 0;
}
}
if ( proccess_list(chan,result_list,1) ){
sleep(1);
continue;
}
/* print the data that was received */
j=1;
while( !(result_list[j-1].ctrl & BC_EOL) ){
printf("Response to message %d: (len: %d, tsw1: %x, tsw2: %x)\n ",j,result_list[j-1].wc,result_list[j-1].tsw[0],result_list[j-1].tsw[1]);
/* print data */
for (k = 0; k < result_list[j-1].wc; k++){
if ( isalnum(result_list[j-1].data[k]) ){
printf("0x%x (%c)",result_list[j-1].data[k],result_list[j-1].data[k]);
}else{
printf("0x%x (.)",result_list[j-1].data[k]);
}
}
printf("\n");
j++;
}
printf("-----------------------------------------------------\n");
printf("Sleeping 15s\n");
sleep(15);
}
}
#elif defined(BRM_BM_TEST)
/* Bus monitor */
/* =========================================================
task1 BM */
rtems_task task1(
rtems_task_argument unused
)
{
brm_t chan;
struct bm_msg msgs[3];
int i;
int cnt;
int tot;
printf("Starting task 1: BM mode\n");
/* open device */
chan=brmlib_open(B1553BRM_DEVICE_NAME);
if ( !chan ){
printf("Failed to open device driver 0\n");
rtems_task_delete(RTEMS_SELF);
return;
}
/* before starting set up
* ¤ blocking mode
* ¤ RT mode
*/
/* Set BM mode */
printf("Task1: Setting BM mode\n");
brmlib_set_mode(chan,BRM_MODE_BM);
/* total blocking mode */
printf("Task1: Setting TX and RX blocking mode\n");
brmlib_set_block(chan,1,1);
tot=0;
while(1){
cnt = brmlib_bm_recv_multiple(chan,msgs,3);
/* print messages */
for (i=0; i<cnt; i++,tot++){
print_bm_msg(tot,&msgs[i]);
}
}
}
#else /* RT mode */
/* =========================================================
task1 RT */
#define MSG_CNT 3
rtems_task task1(
rtems_task_argument unused
)
{
brm_t chan;
struct rt_msg msgs[MSG_CNT];
int i;
char buf[1024];
int ofs,ret;
int msglen;
printf("Starting task 1: RT mode\n");
/* open device */
chan=brmlib_open(B1553BRM_DEVICE_NAME);
if ( !chan ){
printf("Failed to open device driver 0\n");
rtems_task_delete(RTEMS_SELF);
return;
}
/* before starting set up
* ¤ blocking mode
* ¤ RT mode
*/
/* Set RT mode */
printf("Task1: Setting RT mode\n");
brmlib_set_mode(chan,BRM_MODE_RT);
/* total blocking mode */
printf("Task1: Setting TX blocking RX non-blocking mode\n");
brmlib_set_block(chan,1,0);
/* Set RT address (defaults to 1) */
while (1) {
/* read 1 message a time */
ret = brmlib_rt_recv(chan,msgs);
if ( ret <= 0 ){
/* */
printf("Sleeping 1s, %d\n",ret);
sleep(1);
continue;
}
if ( msgs[0].desc >= 32 ){
printf("Message desc >= 32\n");
sleep(1);
continue;
}
printf("--------------- RT MESSAGE: -------------------\n");
if (msgs[0].miw & (1<<7)) {
printf("Message error, desc: %d, miw:%x\n", msgs[0].desc, msgs[0].miw);
}
printf("desc: %d, miw: %x, time: %x\n",msgs[0].desc, msgs[0].miw, msgs[0].time);
msglen = (msgs[0].miw >> 11) & 0x1f;
ofs=0;
for(i=0; i<msglen; i++){
ofs += sprintf(buf+ofs,"0x%04x ",msgs[0].data[i]);
}
printf("Data: %s\n",buf);
printf("------------------------------------------------\n");
/* reply with twisted data */
for(i=0; i<msglen; i++)
msgs[0].data[i] = msgs[0].data[i]+i;
/* respond to incoming message by putting it into
* transmit sub address.
*/
msgs[0].desc += 32;
if ( brmlib_rt_send(chan,msgs) != 1 ){
printf("Error replying\n");
}
}
}
#endif