【Orangepi Zero2 全志H616】驱动舵机控制 / Linux定时器(signal、setitimer)

本文主要是介绍【Orangepi Zero2 全志H616】驱动舵机控制 / Linux定时器(signal、setitimer),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、SG90舵机开发
舵机基本介绍
二、Linux定时器

  • signal 函数
  • setitimer 函数原型
  • signal、setitimer函数API调用

三、舵机 软件PWM实现

一、SG90舵机开发

舵机基本介绍

如下图所示,最便宜的舵机sg90,常用三根或者四根接线,黄色为PWM信号控制用处:
垃圾桶项目开盖用、智能小车的全比例转向、摄像头云台、机械臂等
常见的有0-90°、0-180°、0-360°
在这里插入图片描述
怎么控制转角
向黄色信号线“灌入”PWM信号。
PWM波的频率不能太高,50hz,即周期=1/频率=1/50=0.02s,20ms左右数据:
不同的PWM波形对应不同的旋转角度,以20ms为周期,50hz为频率的PWM波
在这里插入图片描述

二、Linux定时器

signal 函数

signal 函数是一个用于设置信号处理函数的C标准库函数,可用于捕获和处理信号。信号是在UNIX和类UNIX操作系统中用于通知进程发生了特定事件的一种异步通知机制。signal 函数的原型如下:

#include <signal.h>void (*signal(int signum, void (*handler)(int)))(int);

参数说明:

  • signum:要捕获和处理的信号的编号,它是一个整数,如 SIGINT(中断信号)、SIGTERM(终止信号)等。
  • handler:一个函数指针,指向你自己编写的信号处理函数,用于处理接收到的信号。

signal 函数允许你为指定的信号注册一个自定义的处理函数,该处理函数会在程序接收到相应的信号时被调用。处理函数的原型如下:

void handler_function(int signum);

示例用法:

#include <stdio.h>
#include <stdlib.h>
#include <signal.h>// 自定义信号处理函数
void sigint_handler(int signum) {printf("Received SIGINT (Ctrl+C) signal. Exiting...\n");exit(0);
}int main() {// 注册自定义信号处理函数,用于捕获SIGINT信号(Ctrl+C)signal(SIGINT, sigint_handler);printf("Press Ctrl+C to trigger the custom signal handler.\n");while (1) {// 无限循环等待信号}return 0;
}

在上述示例中,signal 函数用于注册一个自定义的信号处理函数 sigint_handler,用于捕获 SIGINT 信号(通常由Ctrl+C触发)。当用户按下Ctrl+C时,程序会执行自定义处理函数,然后退出。这使你能够在程序中捕获和处理特定的信号,以采取适当的行动。不同的信号有不同的用途,可以用于控制程序的行为。

setitimer 函数原型

分析:实现定时器,通过itimerval结构体以及函数setitimer产生的信号,系统随之使用signal信号处理函数
来处理产生的定时信号。从而实现定时器。

setitimer 是一个UNIX系统上的系统调用函数,用于设置和管理定时器。它通常用于定期触发信号或执行某些操作。setitimer 函数的原型如下:

#include <sys/time.h>int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value);

参数说明:

  • which:指定要设置的定时器类型,可以是 ITIMER_REALITIMER_VIRTUALITIMER_PROF 中的一个。
    • ITIMER_REAL:以实际时间计时,通常用于实现定时器功能。
    • ITIMER_VIRTUAL:以进程的虚拟时间(CPU时间)计时。
    • ITIMER_PROF:以进程的虚拟时间和系统时间(CPU和墙钟时间)计时。
  • new_value:一个 struct itimerval 结构,用于指定新的定时器值。
  • old_value:一个 struct itimerval 结构,用于存储旧的定时器值(可选参数)。

struct itimerval 结构定义如下:

struct itimerval {struct timeval it_interval;  // 定时器重复的间隔时间struct timeval it_value;     // 定时器的初始值
};

setitimer 允许你设置定时器的初始值和重复间隔。一旦定时器启动,它将在到期时触发一个信号,通常是 SIGALRM 信号。你可以捕获这个信号并执行相应的操作。

示例用法:

#include <stdio.h>
#include <signal.h>
#include <sys/time.h>void timer_handler(int signum) {// 定时器到期时触发的处理函数printf("Timer expired! (Signum: %d)\n", signum);
}int main() {struct itimerval timer;timer.it_value.tv_sec = 1;  // 初始定时器值为1秒timer.it_value.tv_usec = 0;timer.it_interval.tv_sec = 2;  // 重复间隔为2秒timer.it_interval.tv_usec = 0;// 设置定时器并关联处理函数signal(SIGALRM, timer_handler);setitimer(ITIMER_REAL, &timer, NULL);// 让程序保持运行while (1) {}return 0;
}

上述示例设置了一个每隔1秒触发一次的定时器,并在定时器到期时触发 SIGALRM 信号,执行 timer_handler 处理函数。你可以根据需要修改定时器的初始值和重复间隔以实现不同的定时任务。

signal、setitimer函数API调用

/*该代码实现的功能是: 1s后开启定时器,然后每隔1s向终端打印hello。*/ 
#include <stdio.h> 
#include <sys/time.h> 
#include <stdlib.h> 
#include <signal.h> static int i; void signal_handler(int signum) 
{ i++; if (i == 2000){ printf("hello\n"); i = 0; } 
}int main() 
{ struct itimerval itv; //设定定时时间 itv.it_interval.tv_sec = 0; itv.it_interval.tv_usec = 500;//设定开始生效,启动定时器的时间itv.it_value.tv_sec = 1;itv.it_value.tv_usec = 0;//设定定时方式 if (setitimer(ITIMER_REAL, &itv, NULL) == -1){ perror("error"); exit(-1); }//信号处理 signal(SIGALRM, signal_handler); while(1); return 0; 
}

在这里插入图片描述
这种方法需要注意的是,一个进程只能创建一个定时器

三、舵机 软件PWM实现

SG90编程实现:键盘输入不同的值,让舵机转动,软件PWM实现

#include <stdio.h>
#include <sys/time.h>
#include <stdlib.h>
#include <signal.h>
#include <wiringPi.h>#define SG90Pin 5int angle;
static int i = 0;void signal_handler(int signum)
{if(i <= angle){digitalWrite(SG90Pin, HIGH);}else{digitalWrite(SG90Pin, LOW);}if (i == 40){i = 0;}i++;
}int main()
{struct itimerval itv;angle = 0;wiringPiSetup();pinMode(SG90Pin, OUTPUT);//设定定时时间0.5msitv.it_interval.tv_sec = 0;itv.it_interval.tv_usec = 500;//设定开始生效,启动定时器的时间itv.it_value.tv_sec = 1;itv.it_value.tv_usec = 0;//设定定时方式if (setitimer(ITIMER_REAL, &itv, NULL) == -1){perror("error");exit(-1);}//信号处理signal(SIGALRM, signal_handler);while (1){printf("input angle: 1-0 2-45 3-90 4-135 5-180.\n");scanf("%d", &angle);}return 0;
}

在这里插入图片描述

转圈圈

这篇关于【Orangepi Zero2 全志H616】驱动舵机控制 / Linux定时器(signal、setitimer)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/357140

相关文章

Linux之systemV共享内存方式

《Linux之systemV共享内存方式》:本文主要介绍Linux之systemV共享内存方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、工作原理二、系统调用接口1、申请共享内存(一)key的获取(二)共享内存的申请2、将共享内存段连接到进程地址空间3、将

快速修复一个Panic的Linux内核的技巧

《快速修复一个Panic的Linux内核的技巧》Linux系统中运行了不当的mkinitcpio操作导致内核文件不能正常工作,重启的时候,内核启动中止于Panic状态,该怎么解决这个问题呢?下面我们就... 感谢China编程(www.chinasem.cn)网友 鸢一雨音 的投稿写这篇文章是有原因的。为了配置完

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络

Linux命令之firewalld的用法

《Linux命令之firewalld的用法》:本文主要介绍Linux命令之firewalld的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux命令之firewalld1、程序包2、启动firewalld3、配置文件4、firewalld规则定义的九大

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停