-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.c
182 lines (158 loc) · 5.46 KB
/
main.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
#include <stdint.h>
#include <pru_cfg.h>
#include <pru_intc.h>
#include <pru_rpmsg.h>
#include <pru_i2c_driver.h>
#include <pru_hmc5883l_driver.h>
#include <MPU6050.h>
#include <pru_mpu6050_driver.h>
#include "resource_table.h"
volatile register unsigned __R31;
#define VIRTIO_CONFIG_S_DRIVER_OK 4
#define PAYLOAD_CONTENT_OFFSET 2
unsigned char payload[RPMSG_BUF_SIZE];
uint8_t active = 0;
void initBuffers()
{
{
int i = 0;
for (i = 0; i < RPMSG_BUF_SIZE; i++)
{
payload[i] = '\0';
}
}
}
struct pru_rpmsg_transport transport;
unsigned short src, dst, len;
uint8_t sendData(uint8_t deviceNumber, uint8_t dataBytes, unsigned char* data) {
if(active) {
payload[0] = 'H';
payload[1] = 'D';
pru_rpmsg_send(&transport, dst, src, data - PAYLOAD_CONTENT_OFFSET, dataBytes + PAYLOAD_CONTENT_OFFSET);
}
return 1; // FIXME: da adeguare alla risposta di pru_rpmsg_send
}
uint8_t sendTestData(uint8_t deviceNumber, uint8_t dataBytes, unsigned char* data) {
if(active) {
payload[0] = 'H';
payload[1] = 'S';
pru_rpmsg_send(&transport, dst, src, data - PAYLOAD_CONTENT_OFFSET, dataBytes + PAYLOAD_CONTENT_OFFSET);
}
return 1;
}
HMC5883LConf hmc5883lConf2 = {
payload + PAYLOAD_CONTENT_OFFSET,
payload + PAYLOAD_CONTENT_OFFSET,
sendData,
sendTestData
};
typedef struct {
int16_t ax;
int16_t ay;
int16_t az;
int16_t gx;
int16_t gy;
int16_t gz;
} MPU6450Motion6;
uint8_t pulseHMC5883L() {
if (pru_hmc5883l_driver_Detect(2))
{
if (pru_hmc5883l_driver_Enable(2))
{
if(!pru_hmc5883l_driver_Pulse(2)) {
payload[0] = 'H';
payload[1] = 'F';
payload[1] = 'D';
pru_rpmsg_send(&transport, dst, src, payload, 3);
}
}
else
{
pru_rpmsg_send(&transport, dst, src, "HTK", 4);
}
}
else
{
payload[0] = 'T';
payload[1] = 'F';
pru_rpmsg_send(&transport, dst, src, payload,
PAYLOAD_CONTENT_OFFSET);
}
return 1;
}
uint8_t executeTasks() {
return pulseHMC5883L();
}
uint8_t checkMessages() {
return 1;
}
/**
* main.c
*/
int main(void)
{
volatile unsigned char *status;
/* Allow OCP master port access by the PRU so the PRU can read external memories */
CT_CFG.SYSCFG_bit.STANDBY_INIT = 0;
/* Clear the status of the PRU-ICSS system event that the ARM will use to 'kick' us */
CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST;
/* Make sure the Linux drivers are ready for RPMsg communication */
/* this is another place where a hang could occur */
status = &resourceTable.rpmsg_vdev.status;
while (!(*status & VIRTIO_CONFIG_S_DRIVER_OK))
;
/* Initialize the RPMsg transport structure */
/* this function is defined in rpmsg_pru.c. It's sole purpose is to call pru_virtqueue_init twice (once for
vring0 and once for vring1). pru_virtqueue_init is defined in pru_virtqueue.c. It's sole purpose is to
call vring_init. Not sure yet where that's defined, but it appears to be part of the pru_rpmsg iface.*/
/* should probably test for RPMSG_SUCCESS. If not, then the interface is not working and code should halt */
pru_rpmsg_init(&transport, &resourceTable.rpmsg_vring0,
&resourceTable.rpmsg_vring1, TO_ARM_HOST, FROM_ARM_HOST);
/* Create the RPMsg channel between the PRU and ARM user space using the transport structure. */
// In a real-time environment, rather than waiting forever, this can probably be run loop-after-loop
// until success is achieved. At that point, set a flag and then enable the send/receive functionality
while (pru_rpmsg_channel(RPMSG_NS_CREATE, &transport, CHAN_NAME, CHAN_DESC,
CHAN_PORT) != PRU_RPMSG_SUCCESS)
;
uint32_t counter = 0;
MPU6450Motion6* motion6 = (MPU6450Motion6*)(payload + PAYLOAD_CONTENT_OFFSET);
while (1)
{
if (active)
{
uint8_t ready = pru_mpu6050_driver_GetIntDataReadyStatus();
if(ready) {
payload[0] = 'M';
payload[1] = '6';
pru_mpu6050_driver_GetMotion6(&(motion6->ax), &(motion6->ay), &(motion6->az), &(motion6->gx), &(motion6->gy), &(motion6->gz));
pru_rpmsg_send(&transport, dst, src, payload, sizeof(MPU6450Motion6)+ PAYLOAD_CONTENT_OFFSET);
}
// executeTasks();
}
if (__R31 & HOST_INT)
{
if (pru_rpmsg_receive(&transport, &src, &dst, payload,
&len) == PRU_RPMSG_SUCCESS)
{
if (payload[0] == 'S' && payload[1] == 'T')
{
active = 1;
pru_rpmsg_send(&transport, dst, src, "ST", 3);
if(pru_mpu6050_driver_TestConnection()) {
pru_mpu6050_driver_Initialize();
} else {
pru_rpmsg_send(&transport, dst, src, "MK", 3);
active = 0;
}
} else {
active = 0;
pru_rpmsg_send(&transport, dst, src, "TS", 3);
}
}
else
{
CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST;
}
}
}
}