V3s_ircut.c
13.8 KB
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
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
#include "ircut.h"
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdint.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include "version_choose.h"
#include "JZsdkLib.h"
typedef struct
{
unsigned int CFG[4];
unsigned int DAT;
unsigned int DRV0;
unsigned int DRV1;
unsigned int PUL0;
unsigned int PUL1;
} PIO_Struct;
typedef struct
{
PIO_Struct Pn[7];
} PIO_Map;
typedef enum {
PA = 0,
PB = 1,
PC = 2,
PD = 3,
PE = 4,
PF = 5,
PG = 6,
} PORT;
typedef enum {
IN = 0x00,
OUT = 0x01,
AUX = 0x02,
INT = 0x06,
DISABLE = 0x07,
} PIN_MODE;
unsigned int fd;
PIO_Map *PIO = NULL;
unsigned int addr_start, addr_offset;
unsigned int PageSize, PageMask;
static int PowerContorl;
#define PIO_BASE_ADDR 0x01C20000
#define PIO_ADDR_OFF 0x800
#define PIO_CFG_OFF 0x24 //配置
#define PIO_DAT_OFF 0x34 //数据
#define PWM_ADDR_OFF 0x1400
#define PWM_CH1_OFF 0x08 //pwm1
#define Page_Size (4096 * 2)
#define PIO_BASE_ADDRESS 0x01C20800
#define STATUS_LIGHT_PIN PG
#define STATUS_LIGHT_PIN_NUM 2
static uint32_t *base_map = NULL;
static uint32_t *gpio_map = NULL;
static uint32_t *gpio_cfg = NULL;
static uint32_t *gpio_dat = NULL;
static uint32_t *pwm_base_map = NULL;
static uint32_t *pwm1_period = NULL;
static void GPIO_ConfigPin(PORT port, unsigned int pin, PIN_MODE mode)
{
if (base_map == NULL)
return;
PIO->Pn[port].CFG[pin / 8] &= ~((unsigned int)0x07 << pin % 8 * 4);
PIO->Pn[port].CFG[pin / 8] |= ((unsigned int)mode << pin % 8 * 4);
printf("struct PIO_Struct size : %d\n",sizeof(PIO->Pn[port]));
}
static void GPIO_SetPin(PORT port, unsigned int pin, unsigned int level)
{
if (base_map == NULL)
return;
if (level)
PIO->Pn[port].DAT |= (1 << pin);
else
PIO->Pn[port].DAT &= ~(1 << pin);
}
int GPIO_Free(void)
{
if ((munmap(gpio_map, PageSize * 2)) == 0)//取消映射
{
printf("unmap success!");
}
else
{
printf("unmap failed!");
}
return 0;
}
int V3s_PWM_contrl(int PWM_num)
{
/*设置活跃周期 活跃周期要小于总周期1000*/
(*pwm1_period) &= ~((uint32_t)65535 << 0); //将15~0位置零
(*pwm1_period) |= (uint32_t)PWM_num << 0; //将15~0位置为2500 现在设置活跃周期为35535
/*使能pwm,PWM_CH1_EN为enable*/
(*pwm_base_map) &= ~((uint32_t)1 << 19); //先将第4位置0 ---> disable
(*pwm_base_map) |= (uint32_t)1 << 19; //将第4位置1 ---> enable pwm1
//printf("V3s_Pwm_control:%d\n",PWM_num);
return 0;
}
static void GPIO_mmap_Init()
{
if ((fd = open("/dev/mem", O_RDWR)) == -1)
{
printf("open error");
return;
}
PageSize = sysconf(_SC_PAGESIZE); //使用sysconf查询系统页面大小
PageMask = (~(PageSize - 1)); //页掩码
printf("PageSize:%d,PageMask:0x%.8X", PageSize, PageMask);
addr_start = PIO_BASE_ADDRESS & PageMask; //0x01C20800 & 0xfffff000 = 0x1C20000
addr_offset = PIO_BASE_ADDRESS & ~PageMask; //0x01C20800 & 0x00000100 = 0x800
printf("addr_start:%.8X,addr_offset:0x%.8X\n", addr_start, addr_offset);
//mmap(系统自动分配内存地址,映射区长度“内存页的整数倍”,选择可读可写,MAP_SHARED=与其他所有映射到这个对象的进程共享空间,文件句柄,被映射内容的起点)
//offest 映射物理内存的话,必须页对其!!! 所以这个起始地址应该是0x1000的整数倍,那么明显0x01C20800需要减去0x800才是整数倍!
if ((base_map = mmap(NULL, PageSize * 2, PROT_READ | PROT_WRITE, MAP_SHARED, fd, addr_start)) == NULL)
{
printf("mmap error");
close(fd);
return;
}
printf("base_map:%.8X\n", base_map);
close(fd); //映射好之后就可以关闭设备号
}
// //状态灯初始化
// static void Status_light_init()
// {
// //状态灯初始化
// //这里已经将0x1c20000的地址映射到了内存中,但是我们需要的地址是0x01C20800,所以要再加上地址偏移量~
// PIO = (PIO_Map *)((unsigned int)base_map + addr_offset);
// printf("PIO:0x%.8X\n", PIO);
// GPIO_ConfigPin(STATUS_LIGHT_PIN,STATUS_LIGHT_PIN_NUM,OUT);
// }
// static void Status_light()
// {
// //低电平亮
// GPIO_SetPin(STATUS_LIGHT_PIN,STATUS_LIGHT_PIN_NUM,0);
// }
static void PWM_init() //pwm初始化
{
//pwm引脚初始化
//这里已经将0x1c20000的地址映射到了内存中,但是我们需要的地址是0x01C20800,所以要再加上地址偏移量~
gpio_map = (uint32_t)base_map + PIO_ADDR_OFF; //加上偏移量必选先把他转化成unsigend int才可以相加
//printf("gpio_map:0x%.8X\n", (uint32_t)gpio_map);
gpio_cfg = (uint32_t)gpio_map + PIO_CFG_OFF; //gpioB控制寄存器地址
/**
* PWM波配置顺序
* 1.GPIO 配置PWM输出模式
* 2.PWM 预分頻
* 3.PWM 总周期
* 4.PWM 活跃周期
* 5.PWM 使能
*/
//PB5设定PWM输出
(*gpio_cfg) &= ~((unsigned int)7 << 20); //置20:22 000
(*gpio_cfg) |= ((unsigned int)2 << 20); //PB5 2=010 设定20:22 010 pwm1模式
//我们需要的地址是0x01C21400,所以要再加上地址偏移量
//无法修改为 pwm_base_map = (uint32_t *)base_map + PWM_ADDR_OFF
pwm_base_map = (uint32_t)base_map + PWM_ADDR_OFF; //pwm_base_map也是pwm控制寄存器 初始值是0x00000000
//printf("pwm_base_map:0x%.8X\n", (uint32_t)pwm_base_map);
/*首先设置pwm1 预分頻*/ //PWM_CH1_PRESCAL
(*pwm_base_map) &= ~((uint32_t)15 << 15); //先将15~18位置0
(*pwm_base_map) |= (uint32_t)0 << 15; //将15~18位设置为 0000 ---> 对应分頻120 24m/120=200k
//(*pwm_base_map) |= (uint32_t)9 << 15; //将15~18位设置为 1001 ---> 对应分頻120 24m/24k=1000
/*可能要设置SCLK_CH1_GATING为mask*/
(*pwm_base_map) &= ~((uint32_t)1 << 21); //先将第21位置0
(*pwm_base_map) |= (uint32_t)1 << 21; //将第21位置1 ---> 设置为自定义预分頻系数
/**
* 具体的总周期时间的作用需要进一步测试
* 注意:要活动周期设置好之后使能PWM通道WWW
* 这样才会有正确输出,并且之后直接修改寄存器的值就可以修改占空比。
* */
/*再设置pwm1占空比*/
//无法修改为 pwm1_period = (uint32_t *)pwm_base_map + PWM_CH1_OFF
pwm1_period = (uint32_t)pwm_base_map + PWM_CH1_OFF; //pwm1_period设置pwm_CH1的占空比寄存器
//printf("pwm1_period:0x%.8X\n", (uint32_t)pwm1_period);
/*先设置总周期*/ //PWM周期的计算应该是这样 OSC 24MHz / Pre-scalar / (entire cycles + 1)
(*pwm1_period) &= ~((uint32_t)65535 << 16); //将31~16位置零
(*pwm1_period) |= (uint32_t)1000 << 16; // 现在设置整个周期1000
printf("v3spwm初始化完成\n");
}
// void V3s_Ircut_Init()//引脚初始化
// {
// GPIO_mmap_Init(); //GPIO 内存映射初始化
// //这里已经将0x1c20000的地址映射到了内存中,但是我们需要的地址是0x01C20800,所以要再加上地址偏移量~
// PIO = (PIO_Map *)((unsigned int)base_map + addr_offset);
// printf("PIO:0x%.8X", PIO);
// //静音引脚初始化
// GPIO_ConfigPin(AMPLIFIER_PIN,AMPLIFIER_PIN_NUM,OUT);
// //10T要初始化电机引脚
// if (DEVICE_VERSION == JZ_H10T)
// {
// PWM_init() ; //pwm初始化
// GPIO_ConfigPin(PB,6,OUT); //对外供电初始化
// }
// //激光引脚初始化
// if (DEVICE_VERSION == JZ_U3S)
// {
// GPIO_ConfigPin(PB,6,OUT);
// GPIO_ConfigPin(PB,7,OUT);
// }
// //状态灯初始化
// //Status_light_init();
// //打开状态灯
// //Status_light();
// }
int V3s_Ircut_uInit()
{
//GPIO 内存映射移除
munmap(base_map, Page_Size);
printf("munmap success!\n");
}
// void Ircut_V3S_Set_OutPutPowerControl(int power_control)
// {
// if (DEVICE_VERSION == JZ_H10T)
// {
// if (power_control == 1)
// {
// GPIO_ConfigPin(PB,6,OUT); //电源脚初始化
// GPIO_SetPin(PB,6,1); //打开开关
// }
// else if(power_control ==0)
// {
// GPIO_SetPin(PB,6,0); //关闭开关
// GPIO_ConfigPin(PB,6,DISABLE); //释放引脚
// }
// }
// }
/********************
*
* 端口组换算
* 将JZSDK的端口组转换成V3S的端口组
*
* **********************/
static T_JZsdkReturnCode V3s_Port_Conversion(E_JZ_IRC_PORT port, PORT *v3s_port)
{
switch (port)
{
case JZ_IRC_PORT_PA:
*v3s_port = PA;
break;
case JZ_IRC_PORT_PB:
*v3s_port = PB;
break;
case JZ_IRC_PORT_PC:
*v3s_port = PC;
break;
case JZ_IRC_PORT_PD:
*v3s_port = PD;
break;
case JZ_IRC_PORT_PE:
*v3s_port = PE;
break;
case JZ_IRC_PORT_PF:
*v3s_port = PF;
break;
case JZ_IRC_PORT_PG:
*v3s_port = PG;
break;
default:
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
break;
}
return JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS;
}
/********************
*
* 端口模式换算
* 将JZSDK的端口模式转换成V3S的端口模式
*
* **********************/
static T_JZsdkReturnCode V3s_PinMode_Conversion(E_JZ_IRC_PIN_MODE mode, PIN_MODE *v3s_mode)
{
switch (mode)
{
case JZ_IRC_PIN_MODE_IN:
*v3s_mode = IN;
break;
case JZ_IRC_PIN_MODE_OUT:
*v3s_mode = OUT;
break;
case JZ_IRC_PIN_MODE_AUX:
*v3s_mode = AUX;
break;
case JZ_IRC_PIN_MODE_INT:
*v3s_mode = INT;
break;
case JZ_IRC_PIN_MODE_DISABLE:
*v3s_mode = DISABLE;
break;
default:
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
break;
}
return JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS;
}
T_JZsdkReturnCode JZsdk_V3s_Ircut_Init(IrcutInfo *ircut_info)
{
//内存映射初始化
GPIO_mmap_Init();
//地址偏移量
//这里已经将0x1c20000的地址映射到了内存中,但是我们需要的地址是0x01C20800,所以要再加上地址偏移量~
PIO = (PIO_Map *)((unsigned int)base_map + addr_offset);
printf("PIO:0x%.8X", PIO);
//静音角引脚初始化
if (ircut_info->Amplifiter_Pin.Enable == JZ_FLAGCODE_ON)
{
//获取引脚组
PORT Amplifiter_Pin_port;
if(V3s_Port_Conversion(ircut_info->Amplifiter_Pin.Port, &Amplifiter_Pin_port) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_Port_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//获取引脚模式
PIN_MODE Amplifiter_Pin_mode;
if(V3s_PinMode_Conversion(ircut_info->Amplifiter_Pin.mode, &Amplifiter_Pin_mode) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_PinMode_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//配置引脚
GPIO_ConfigPin(Amplifiter_Pin_port,ircut_info->Amplifiter_Pin.index,Amplifiter_Pin_mode);
}
if (ircut_info->SideLight_Right_Pin.Enable == JZ_FLAGCODE_ON)
{
//获取引脚组
PORT SideLight_Right_Pin_port;
if(V3s_Port_Conversion(ircut_info->SideLight_Right_Pin.Port, &SideLight_Right_Pin_port) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_Port_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//获取引脚模式
PIN_MODE SideLight_Right_Pin_mode;
if(V3s_PinMode_Conversion(ircut_info->SideLight_Right_Pin.mode, &SideLight_Right_Pin_mode) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_PinMode_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//配置引脚
GPIO_ConfigPin(SideLight_Right_Pin_port,ircut_info->SideLight_Right_Pin.index,SideLight_Right_Pin_mode);
}
if (ircut_info->SideLight_Left_Pin.Enable == JZ_FLAGCODE_ON)
{
//获取引脚组
PORT SideLight_Left_Pin_port;
if(V3s_Port_Conversion(ircut_info->SideLight_Left_Pin.Port, &SideLight_Left_Pin_port) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_Port_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//获取引脚模式
PIN_MODE SideLight_Left_Pin_mode;
if(V3s_PinMode_Conversion(ircut_info->SideLight_Left_Pin.mode, &SideLight_Left_Pin_mode) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_PinMode_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//配置引脚
GPIO_ConfigPin(SideLight_Left_Pin_port,ircut_info->SideLight_Left_Pin.index,SideLight_Left_Pin_mode);
}
if (ircut_info->PWM_1_Pin.Enable == JZ_FLAGCODE_ON)
{
//配置PWM
PWM_init();
}
return JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS;
}
/***************************
*
* 设置引脚状态
*
*
* ****************************/
T_JZsdkReturnCode JZsdk_V3s_Gpio_SetPin(E_JZ_IRC_PORT port, int index, int status)
{
//转换引脚组
PORT gpio_port;
if(V3s_Port_Conversion(port, &gpio_port) != JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS)
{
JZSDK_LOG_ERROR("V3s_Port_Conversion error!\n");
return JZ_ERROR_SYSTEM_MODULE_CODE_FAILURE;
}
//转换引脚状态
int value = 0;
if (status == JZ_FLAGCODE_ON)
{
value = 1;
}
else if (status == JZ_FLAGCODE_OFF)
{
value = 0;
}
//设置引脚状态
GPIO_SetPin(gpio_port, index, value);
return JZ_ERROR_SYSTEM_MODULE_CODE_SUCCESS;
}