linux下一个iic驱动(按键+点灯)-互斥

news2025/1/11 21:04:12

一、前提:

硬件部分:

1. rk3399开发板,其中的某一路iic,这个作为总线的主控制器

2. gd32单片机,其中的某一路iic,从设备。主要是按键上报和灯的亮灭控制。(按键大约30个,灯在键的下面,对应有30个左右。)

3. iic主要的功能是两个部分,是主动上报按键值,和接收点灯命令(收到后需要发送应答)。

4. 该路iic总线上还有一个触摸屏的设备。(不在介绍范围,这个说明一下单片机端不能改为主设备,cpu改为从设备的原因)。

5.cpu与单片机之间有一个中断io引脚,用于有按键(或者应答)数据时,提示cpu(rk3399)的作用,低电平有效。

软件部分:

1.linux4.4.194的内核,arm64平台,自编ic驱动程序。

2.单片机也是自己做的iic程序,使用硬件iic,协议自定义,与驱动一致即可。(本文不介绍单片机的程序)。

3.数据帧的格式(前期已经定好,不再修改):cpu(rk3399,下同)发送指令由5个字节组成,设备(单片机,下同)回复指令由7个字节组成。

cpu主动发送部分:

 设备发送部分:

 

二、iic驱动基本思路和逻辑

1. 发送数据时,首先把数据组合好,调用i2c_master_send,完成发送。(比如点亮熄灭的命令)

i2c_master_send(s_jc_keyboard_info.i2c_client, (char *)&keyboard_send_msg, sizeof(KEYBOARD_I2C_SEND_MSG_S)) != sizeof(KEYBOARD_I2C_SEND_MSG_S)

2.接收数据,单片机发送数据时,是肯定有一个中断触发cpu,所以cpu处理是在中断处理函数中。

   收到中断后,考虑到iic的数据是串行通信,比较费时,所以采用工作队列的方式处理,收到中断,其实就是启动一次工作队列,这时使用了内核自己的队列。

    队列启动后,使用i2c_master_recv接收数据,然后根据协议解析数据,完成相关的操作。

static irqreturn_t s_jc_keyboard_i2c_isr(int irq, void *dev_id)
{
	if(irq != s_jc_keyboard_info.i2c_client->irq) {
		return IRQ_NONE;
	}

	if(!schedule_work(&jc_keyboard_work)) {
		pr_err("Error schedule_work!\n");
	}
	
	return IRQ_HANDLED;
}
i2c_master_recv(s_jc_keyboard_info.i2c_client, (char *)&keyboard_recv_msg, sizeof(KEYBOARD_I2C_RECV_MSG_S)) != sizeof(KEYBOARD_I2C_RECV_MSG_S)

三、一个互斥问题。

1. 有闪烁灯的控制,而且是多个灯的闪烁,频率

    这种情况,cpu主动发送数据和设备应答的数据(cpu主动读取数据)比较多,每一次设备应答都会中断,中断比较频繁。

2.同时多次按键,组合键

    这个情况下,中断比较多,cpu主要是读取数据。

3.确实有异常的表现,报出一些数据帧不能解析的错误,偶尔有按键不能正常上报按下和松开的状态。

4.客户在使用的时候,发现多个灯闪的时候,会出现比较大的卡顿现象(后来分析其实是命令丢失,就是第一个点亮或熄灭命令发了很久之后,才开始闪),同时按键也会出现失灵的现象(可能要多次按键才会作用)。

5. 不知道是不是频繁中断的影响,schedule_work偶尔也会报出错误提示,原因未知。

四、解决思路

1. 中断是个问题,cpu发送数据和接收数据的过程中是很有可能被中断打断的,这个打断会不会导致数据帧出现错误。尤其是有数据下发的点灯命令和频繁按键上报的情况。

2.点灯命令的应答和按键上报的中断同时(或者间隔很短的情况下)会如何?

3.加上互斥锁(iic_idel_Mutex),一个发送流程(包括发送和接收应答)走完,才能进行下一步操作。比如点灯,必须把数据发完,同时接收到应答(或者超时)才能发送下一帧数据。

4.第二把互斥锁(iic_Mutex,不知道是否多余),每一次发送和接收都加锁,接收发送函数结束后解锁。

5.加上自定义的一个工作队列,使用自己的工作队列处理中断事件。

五、加上锁后,基本解决这个问题。

详细请参考一下源代码(部分主要源码,h文件省略了)

/*
 * jc_keyboard.c
 *
 *  Created on: Dec 25, 2021
 *      Author: dazhi
 */

#include <linux/module.h>
#include <linux/irq.h>
#include <linux/interrupt.h>
#include <linux/fs.h>
#include <linux/i2c.h>
#include <linux/of_gpio.h>
#include <linux/input.h>
#include <linux/miscdevice.h>
#include <linux/workqueue.h>
#include <linux/semaphore.h>
#include <asm/uaccess.h>
#include <linux/moduleparam.h>


#include "jc_keyboard.h"
#include "jc_keyboard_cmd.h"
#include "jc_keyuser.h"

#define JC_KEYBOARD_DRIVER_NAME		"jc_keyboard"
#define JC_KEYBOARD_IRQ_GPIO_NAME	"jc_keyboard_irq_gpio"
#define JC_KEYBOARD_IRQ_NAME		"jc_keyboard_irq"
#define JC_KEYBOARD_MISCDEV_MINOR	(20)
#define JC_KEYBOARD_I2C_TIMEOUT		(4000)	//4000 --> 1s  2022-09-06 dazhi修改,客户处调试暂时已通过。

#define NO_POLL_MODE   //不主动查询,需要单片机给出中断才行。
#define USE_SELF_WORK_QUEUE 

//static int jc_keyboard_inited = -1;  //是否加载成功,2023-01-09

#ifdef USE_SELF_WORK_QUEUE
//分配工作队列的指针
static struct workqueue_struct *jc_keyboard_wq;
#endif

static int debug_print = 0;   //默认不打印调试语句

static struct mutex iic_Mutex;      //互斥锁
static struct mutex iic_idel_Mutex;      //iic空闲互斥锁
static volatile unsigned long wait_respon_bits = 0; //需要等待什么应答,每一个位对应一种需求


module_param(debug_print,int, 0644);   //加载时传递参数

typedef struct {
	int irq_gpio;
	struct input_dev *input_dev;
	struct i2c_client *i2c_client;
	struct semaphore ioctl_sem;
} JC_KEYBOARD_INFO;

static char s_i2c_reply_ret = 0;
static struct miscdevice s_jc_keyboard_miscdevice;
static struct work_struct jc_keyboard_work;
static JC_KEYBOARD_INFO s_jc_keyboard_info;


//按键转换值的数组,定义在jc_keyuser.h
static const unsigned char s_user_key_value[] = {
		0,
		USER_KEY_VALUE_FUNCTION_1,   //0x1
		USER_KEY_VALUE_FUNCTION_2,   //0x2
		USER_KEY_VALUE_FUNCTION_3,   //0x3
		USER_KEY_VALUE_FUNCTION_4,   //0x4
		USER_KEY_VALUE_FUNCTION_5,   //0x5
		USER_KEY_VALUE_FUNCTION_6,   //0x6
		USER_KEY_VALUE_FUNCTION_7,   //0x7
		USER_KEY_VALUE_FUNCTION_8,   //0x8
		USER_KEY_VALUE_FUNCTION_9,   //0x9
		USER_KEY_VALUE_FUNCTION_10,   //0xa
		USER_KEY_VALUE_FUNCTION_11,   //0xb
		USER_KEY_VALUE_FUNCTION_12,   //0xc

		USER_KEY_VALUE_0,    //索引号13  -- 数字0
		USER_KEY_VALUE_1,    //索引号14  -- 数字1
		USER_KEY_VALUE_2,    //索引号15  -- 数字2
		USER_KEY_VALUE_3,    //索引号16  -- 数字3
		USER_KEY_VALUE_4,    //索引号17  -- 数字4
		USER_KEY_VALUE_5,    //索引号18  -- 数字5
		USER_KEY_VALUE_6,    //索引号19  -- 数字6
		USER_KEY_VALUE_7,    //索引号20  -- 数字7
		USER_KEY_VALUE_8,    //索引号21  -- 数字8
		USER_KEY_VALUE_9,    //索引号22  -- 数字9
		USER_KEY_VALUE_ASTERISK,    //索引号23  -- 
		USER_KEY_VALUE_POUND,    //索引号24  -- 

		USER_KEY_VALUE_TELL,    //索引号25
		USER_KEY_VALUE_VOLUME_INCREASE,    //索引号26
		USER_KEY_VALUE_VOLUME_DECREASE,    //索引号27

		USER_KEY_VALUE_SWITCH,
		USER_KEY_VALUE_INSIDE,
		USER_KEY_VALUE_OUTSIDE,

		USER_KEY_VALUE_UP,    //索引号31
		USER_KEY_VALUE_DOWN,    //索引号32
		USER_KEY_VALUE_PTT,    //索引号33
		USER_KEY_VALUE_LEFT,    //索引号34
		USER_KEY_VALUE_RIGHT,    //索引号35
		USER_KEY_VALUE_OK,    //索引号36

		USER_KEY_VALUE_KEEP,    //索引号37
		USER_KEY_VALUE_SET,    //索引号38
		USER_KEY_VALUE_TEST    //索引号39---> 0x27
};


//左侧(中括号中的数)是客户自动自定义的键值 比如KC_L1 --> 1
//右侧是s_user_key_value中对应值得索引号,比如数字0,在数组中排13,就是0xd,
static const unsigned char s_user_map_led_value[] = {
		[1] = 0x01	, //图示1左1	     //KC_L1   1
		[3] = 0x02	, //图示2左2	     //KC_L2   3
		[5] = 0x03	, //图示3左3	     //KC_L3   5
		[7] = 0x04	, //图示4左4	     //KC_L4   7
		[9] = 0x05	, //图示5左5	     //KC_L5   9
		[44] = 0x06	, //图示6左6	     //KC_L6   44
		[2] = 0x07	, //图示7右1	     //KC_R1   2
		[4] = 0x08	, //图示8右2	     //KC_R2   4
		[6] = 0x09	, //图示9右3	     //KC_R3   6
		[8] = 0x0A	, //图示10右4     //KC_R4	8  
		[10] = 0x0B	, //图示11右5     //KC_R5	10
		[45] = 0x0C	, //图示12右6	//KC_R6    45
		[27] = 0x0D	, //0		//KC_NUM0   27
		[18] = 0x0E	, //1		//KC_NUM1   18
		[19] = 0x0F	, //2		//KC_NUM2   19	
		[20] = 0x10	, //3		//KC_NUM3   20	
		[21] = 0x11	, //4		//KC_NUM4   21	
		[22] = 0x12	, //5		//KC_NUM5   22	
		[23] = 0x13	, //6		//KC_NUM6   23	
		[24] = 0x14	, //7		//KC_NUM7   24	
		[25] = 0x15	, //8		//KC_NUM8   25	
		[26] = 0x16	, //9		//KC_NUM9   26	
		[28] = 0x17	, //*		//KC_DOT   28	
		[29] = 0x18	, //#		//KC_CLEAR   29	
		[13] = 0x19	, //电话(拨号)			//KC_TEL   13
		[35] = 0x1A	, //音量+		//KC_VOLUME_UP	   35
		[36] = 0x1B	, //音量-		//KC_VOLUME_DOWN	36	
		[14] = 0x27	, //TEST		//KC_TESKKEY   14   //原0x1c改0x27
		[11] = 0x1D	, //内通			//KC_INC      11
		[12] = 0x1E	, //外通			//KC_EXC      12 
		[30] = 0x1F	, //上			//KC_UP       30
		[31] = 0x20	, //下			//KC_DOWN     31
		[17] = 0x21	, //PTT		//KC_PTT        17	
		[32] = 0x22	, //左			//KC_LEFT   32
		[33] = 0x23	, //右			//KC_FIGHT  33
		[34] = 0x24	, //OK			//KC_OK     34
		[15] = 0x25	, //保持			//KC_HOLD   15
		[43] = 0x26	, //设置/摘挂机		//KC_RECOV  16改成43 原来[16] = 0x26	,
		[42] = 0x27	, //测试/复位		//KC_xxx	
		[37] = 0x28	, //指示灯-红		//LED_RED  37	
		[38] = 0x29	, //指示灯-绿		//LED_GREEN 38	
		[39] = 0x2A	, //指示灯-蓝		//KC_BLUE   39	
		[40] = 0x2B	 //全部键灯			//ALL_KEY_LED  40
};

static const struct of_device_id s_jc_keyboard_match_table[] = {
		{ .compatible = "jc,keyboard", },
		{},
};

static const struct i2c_device_id s_jc_keyboard_i2c_id[] = {
		{ JC_KEYBOARD_DRIVER_NAME, 0},
		{}
};

static void s_jc_keyboard_work_func_t(struct work_struct *work) {
	KEYBOARD_I2C_RECV_MSG_S keyboard_recv_msg;
	unsigned char i = 0;
	unsigned char cmd_verify_tmp = 0;
//	pr_err("2023debug s_jc_keyboard_work_func_t\n");
	memset(&keyboard_recv_msg, 0, sizeof(KEYBOARD_I2C_RECV_MSG_S));
	mutex_lock(&iic_Mutex);  //加锁
	if(i2c_master_recv(s_jc_keyboard_info.i2c_client, (char *)&keyboard_recv_msg, sizeof(KEYBOARD_I2C_RECV_MSG_S)) != sizeof(KEYBOARD_I2C_RECV_MSG_S)) {
		pr_err("Error i2c_master_recv, addr = %#x\n", s_jc_keyboard_info.i2c_client->addr);
		mutex_unlock(&iic_Mutex);  //开锁
		return;
	}
	mutex_unlock(&iic_Mutex);  //开锁

	if(debug_print)
		pr_err("i2c_master_recv: h0 = %#x, h1 = %#x type = %#x, key0=%#x, key1=%#x,key2=%#x,verify=%#x\n",
			keyboard_recv_msg.cmd_header0,
			keyboard_recv_msg.cmd_header1,
			keyboard_recv_msg.cmd_type,
			keyboard_recv_msg.cmd_key0,
			keyboard_recv_msg.cmd_key1,
			keyboard_recv_msg.cmd_key2,
			keyboard_recv_msg.cmd_verify);

	cmd_verify_tmp = FRAME_VERIFY(
			keyboard_recv_msg.cmd_header0,
			keyboard_recv_msg.cmd_header1,
			keyboard_recv_msg.cmd_type,
			keyboard_recv_msg.cmd_key0,
			keyboard_recv_msg.cmd_key1,
			keyboard_recv_msg.cmd_key2);
	if(keyboard_recv_msg.cmd_header0 != FRAME_HEADER_0 || keyboard_recv_msg.cmd_header1 != FRAME_HEADER_1) {
		pr_err("Error recv cmd header: %#x %#x\n", keyboard_recv_msg.cmd_header0, keyboard_recv_msg.cmd_header1);
		return;
	}
	if(keyboard_recv_msg.cmd_verify != cmd_verify_tmp) {
		if(debug_print)
			pr_err("Error verify %#x\n", keyboard_recv_msg.cmd_verify);
		return;
	}

	if((keyboard_recv_msg.cmd_type & 0xfc) == FRAME_CMD_TYPE_KEY_LED_FLASH)  //闪烁的指令
	{
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(KEY_LED_FLASH_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		return ;		
	}

//	pr_err("2023debug s_jc_keyboard_work_func_t cmd_type = %d,cmd_key2 = %d\n",keyboard_recv_msg.cmd_type,keyboard_recv_msg.cmd_key2);
	switch(keyboard_recv_msg.cmd_type) {
	case FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_PRESS:
	case FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_RELEASE:
		if(s_jc_keyboard_info.input_dev) {
		//	unsigned char key;
			unsigned char key_val[3] = {keyboard_recv_msg.cmd_key0,keyboard_recv_msg.cmd_key1,keyboard_recv_msg.cmd_key2};
			if(debug_print)
				pr_info("key1 = %d key2 = %d key3 = %d\n",key_val[0],key_val[1],key_val[2]);
			for(i=0;(i<3);i++)
			//for(i=0;(i<3) && key_val[i];i++)
			{
				//key = keyboard_recv_msg.cmd_key0? keyboard_recv_msg.cmd_key0:keyboard_recv_msg.cmd_key2;
				if(key_val[i] < KEY_VALUE_FUNCTION_1 || key_val[i] > KEY_VALUE_TEST) {
					continue;
					// pr_err("Error key %#x out of range!\n", key_val[i]);
					// break;
				}
				input_report_key(s_jc_keyboard_info.input_dev, s_user_key_value[key_val[i]], (keyboard_recv_msg.cmd_type == FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_PRESS)? 1:0);
			}
			input_sync(s_jc_keyboard_info.input_dev);
		}
		break;
	case FRAME_CMD_TYPE_GET_BRIGHTNESS:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_GET_PANEL_MODEL:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_GET_PANEL_VER:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_SET_BRIGHTNESS:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_RESET:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(RESET_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_KEY_LED_ON:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(KEY_LED_ON_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	case FRAME_CMD_TYPE_KEY_LED_OFF:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
		clear_bit(KEY_LED_OFF_RESPONSE_BIT, &wait_respon_bits);   //清零某一位
		up(&s_jc_keyboard_info.ioctl_sem);
		break;
	default:
		pr_err("Error non-supported cmd %#x\n", keyboard_recv_msg.cmd_type);
		break;
	}
}


#ifndef NO_POLL_MODE
//读取消息超时的处理
static int s_jc_keyboard_inttimeout_func_t(void) {
	KEYBOARD_I2C_RECV_MSG_S keyboard_recv_msg;
	unsigned char cmd_verify_tmp = 0;

	// if(jc_keyboard_inited != 1)  //驱动没有加载成功!!
	// 	return -1;
//	pr_err("2023debug s_jc_keyboard_inttimeout_func_t\n");
	memset(&keyboard_recv_msg, 0, sizeof(KEYBOARD_I2C_RECV_MSG_S));
	mutex_lock(&iic_Mutex);  //加锁
	if(i2c_master_recv(s_jc_keyboard_info.i2c_client, (char *)&keyboard_recv_msg, sizeof(KEYBOARD_I2C_RECV_MSG_S)) != sizeof(KEYBOARD_I2C_RECV_MSG_S)) {
		pr_err("Error i2c_master_recv, addr = %#x\n", s_jc_keyboard_info.i2c_client->addr);
		mutex_unlock(&iic_Mutex);  //开锁
		return -1;
	}
	mutex_unlock(&iic_Mutex);  //开锁

	cmd_verify_tmp = FRAME_VERIFY(
			keyboard_recv_msg.cmd_header0,
			keyboard_recv_msg.cmd_header1,
			keyboard_recv_msg.cmd_type,
			keyboard_recv_msg.cmd_key0,
			keyboard_recv_msg.cmd_key1,
			keyboard_recv_msg.cmd_key2);
	if(keyboard_recv_msg.cmd_header0 != FRAME_HEADER_0 || keyboard_recv_msg.cmd_header1 != FRAME_HEADER_1) {
		pr_err("Error recv cmd header: %#x %#x\n", keyboard_recv_msg.cmd_header0, keyboard_recv_msg.cmd_header1);
		return -1;
	}
	if(keyboard_recv_msg.cmd_verify != cmd_verify_tmp) {
		pr_err("Error verify %#x\n", keyboard_recv_msg.cmd_verify);
		return -1;
	}

	switch(keyboard_recv_msg.cmd_type) {
	case FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_PRESS:
	case FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_RELEASE:
		if(s_jc_keyboard_info.input_dev) {
			unsigned char key = keyboard_recv_msg.cmd_key0? keyboard_recv_msg.cmd_key0:keyboard_recv_msg.cmd_key2;
			if(key < KEY_VALUE_FUNCTION_1 || key > KEY_VALUE_TEST) {
				pr_err("Error key %#x out of range!\n", key);
				break;
			}
			input_report_key(s_jc_keyboard_info.input_dev, s_user_key_value[key], (keyboard_recv_msg.cmd_type == FRAME_CMD_TYPE_GET_KEY_VALUE_REPLY_PRESS)? 1:0);
			input_sync(s_jc_keyboard_info.input_dev);
		}
		break;
	case FRAME_CMD_TYPE_GET_BRIGHTNESS:
	case FRAME_CMD_TYPE_GET_PANEL_MODEL:
	case FRAME_CMD_TYPE_GET_PANEL_VER:
	case FRAME_CMD_TYPE_SET_BRIGHTNESS:
	case FRAME_CMD_TYPE_RESET:
	case FRAME_CMD_TYPE_KEY_LED_ON:
	case FRAME_CMD_TYPE_KEY_LED_OFF:
		s_i2c_reply_ret = keyboard_recv_msg.cmd_key2;
	//	up(&s_jc_keyboard_info.ioctl_sem);
		break;
	default:
		pr_err("Error non-supported cmd %#x\n", keyboard_recv_msg.cmd_type);
		return -1;
	}
	return 0;
}
#endif



static irqreturn_t s_jc_keyboard_i2c_isr(int irq, void *dev_id)
{
	if(irq != s_jc_keyboard_info.i2c_client->irq) {
		return IRQ_NONE;
	}
	if(debug_print)
		pr_err("enter jc_keyboard_i2c_isr\n");

#ifdef USE_SELF_WORK_QUEUE
	//将自己的工作和自己的工作队列进行管理,然后再登记
    if(!queue_work(jc_keyboard_wq, &jc_keyboard_work)) {
		pr_err("Error queue_work 2023!\n");
	}
#else
	if(!schedule_work(&jc_keyboard_work)) {
		pr_err("Error schedule_work!\n");
	}
#endif	
	return IRQ_HANDLED;
}

static int s_jc_keyboard_irq_init(void) {
	if(s_jc_keyboard_info.irq_gpio <= 0) {
		pr_err("Error irq_gpio is %d\n", s_jc_keyboard_info.irq_gpio);
		return -1;
	}
	if(s_jc_keyboard_info.i2c_client->irq <= 0) {
		pr_err("Error irq is %d\n", s_jc_keyboard_info.i2c_client->irq);
		return -1;
	}
	if(gpio_request(s_jc_keyboard_info.irq_gpio, JC_KEYBOARD_IRQ_GPIO_NAME)) {
		pr_err("Error gpio_request %s\n", JC_KEYBOARD_IRQ_GPIO_NAME);
		return -1;
	}
	if(gpio_direction_input(s_jc_keyboard_info.irq_gpio)) {
		pr_err("Error gpio_direction_input %d\n", s_jc_keyboard_info.irq_gpio);
		gpio_free(s_jc_keyboard_info.irq_gpio);
		return -1;
	}
	INIT_WORK(&jc_keyboard_work, s_jc_keyboard_work_func_t);

#ifdef USE_SELF_WORK_QUEUE
	//创建自己的工作队列和自己的内核线程
    jc_keyboard_wq = create_workqueue("myjc_keyboard_wq");
#endif
	if(request_irq(s_jc_keyboard_info.i2c_client->irq, s_jc_keyboard_i2c_isr, IRQF_TRIGGER_FALLING, JC_KEYBOARD_IRQ_NAME, NULL)) {
		pr_err("Error request_irq!\n");
		gpio_free(s_jc_keyboard_info.irq_gpio);
		return -1;
	}
	return 0;
}

static int s_jc_keyboard_irq_exit(void) {
	if(s_jc_keyboard_info.i2c_client->irq > 0) {
		free_irq(s_jc_keyboard_info.i2c_client->irq, NULL);
	}
	if(s_jc_keyboard_info.irq_gpio > 0) {
		gpio_free(s_jc_keyboard_info.irq_gpio);
	}

#ifdef USE_SELF_WORK_QUEUE
	//销毁自己的工作队列和内核线程
    destroy_workqueue(jc_keyboard_wq);
#endif
	return 0;
}

static int s_jc_keyboard_input_init(void) {
	int i = 0;
	if(s_jc_keyboard_info.input_dev) {
		pr_err("Error input_dev is not null!\n");
		return -1;
	}
	if(!(s_jc_keyboard_info.input_dev = input_allocate_device())) {
		pr_err("Error input_allocate_device!\n");
		return -1;
	}

	s_jc_keyboard_info.input_dev->name = JC_KEYBOARD_DRIVER_NAME;
	for(i = KEY_VALUE_FUNCTION_1; i <= KEY_VALUE_TEST; i ++) {
		input_set_capability(s_jc_keyboard_info.input_dev, EV_KEY, s_user_key_value[i]);
	}
	if(input_register_device(s_jc_keyboard_info.input_dev)) {
		pr_err("Error input_register_device!\n");
		input_free_device(s_jc_keyboard_info.input_dev);
		s_jc_keyboard_info.input_dev = NULL;
		return -1;
	}
	return 0;
}

static int s_jc_keyboard_input_exit(void) {
	if(!s_jc_keyboard_info.input_dev) {
		pr_err("Error input_dev is null!\n");
		return -1;
	}
	input_unregister_device(s_jc_keyboard_info.input_dev);
	input_free_device(s_jc_keyboard_info.input_dev);
	s_jc_keyboard_info.input_dev = NULL;
	return 0;
}

static long s_jc_keyboard_unlocked_ioctl(struct file *file, unsigned int cmd, unsigned long argv) {
	KEYBOARD_I2C_SEND_MSG_S keyboard_send_msg;
	if(debug_print)
		pr_err("enter jc_keyboard_unlocked_ioctl\n");
	// if(jc_keyboard_inited != 1)  //驱动没有加载成功!!
	// 	return -1;

	if(_IOC_TYPE(cmd) != KEYBOARD_IOC_MAGIC) {
		pr_err("Error cmd %d\n", cmd);
		return -1;
	}
	memset(&keyboard_send_msg, 0, sizeof(KEYBOARD_I2C_SEND_MSG_S));
	keyboard_send_msg.cmd_header0 = FRAME_HEADER_0;
	keyboard_send_msg.cmd_header1 = FRAME_HEADER_1;

	switch(cmd) {
	case KEYBOARD_IOC_GET_BRIGHTNESS:
		keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_GET_BRIGHTNESS;
		set_bit(BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_SET_BRIGHTNESS:
		if(!argv || copy_from_user(&keyboard_send_msg.cmd, (void *)argv, 1)) {
			pr_err("Error copy_from_user!\n");
			return -2;
		}
		if(keyboard_send_msg.cmd < FRAME_CMD_TYPE_SET_BRIGHTNESS_MIN || keyboard_send_msg.cmd > FRAME_CMD_TYPE_SET_BRIGHTNESS_MAX) {
			pr_err("Error brightness out of range!\n");
			return -3;
		}
		keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_SET_BRIGHTNESS;
		set_bit(SET_BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_GET_PANEL_MODEL:
		keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_GET_PANEL_MODEL;
		set_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_GET_PANEL_VER:
		keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_GET_PANEL_VER;
		set_bit(PANEL_VER_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_RESET:
		keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_RESET;
		set_bit(RESET_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_KEY_LED_FLASH:   //2023-04-28
		{
			unsigned char user_val = 0;
			unsigned char flashtype = 0;

			if(!argv || copy_from_user(&user_val, (void *)argv, 1)) {
				pr_err("Error KEYBOARD_IOC_KEY_LED_FLASH copy_from_user!\n");
				return -4;
			}
			keyboard_send_msg.cmd = s_user_map_led_value[user_val & 0x3f];//(user_val & 0x3f);	//闪烁哪个灯
			flashtype = ((user_val >> 6) & 0x3);   //闪烁类型
			keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_KEY_LED_FLASH | flashtype;  //发过去的命令是0x80,81,82,83
		}
		set_bit(KEY_LED_FLASH_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		break;
	case KEYBOARD_IOC_KEY_LED_ON:
	case KEYBOARD_IOC_KEY_LED_OFF: {
	//	int i = KEY_VALUE_FUNCTION_1;
		unsigned char user_key_code = 0;
		if(!argv || copy_from_user(&user_key_code, (void *)argv, 1)) {
			pr_err("Error copy_from_user!\n");
			return -5;
		}
		// for(; i < KEY_VALUE_TEST; i ++) {
		// 	if(s_user_key_value[i] == user_key_code) {
		// 		keyboard_send_msg.cmd = i;
		// 		break;
		// 	}
		// }
		if(user_key_code < sizeof(s_user_map_led_value)/sizeof(s_user_map_led_value[0]))
			keyboard_send_msg.cmd = s_user_map_led_value[user_key_code];
		else
			keyboard_send_msg.cmd = 0;  //20220905

		//2023-01-05  不能识别的按键
		if(keyboard_send_msg.cmd == 0){
			pr_err("Error keyboard_send_msg.cmd == 0! user_key_code = %d\n",user_key_code);
			return -6;
		}

		// if(i >= KEY_VALUE_TEST) {
		// 	if(user_key_code >= KEY_RED_LED && user_key_code <= ALL_KEY_VALUE) {  /*lsr modify 20220512 (KEY_BLUE_LED)*/
		// 		keyboard_send_msg.cmd = user_key_code;
		// 	}
		// 	else {
		// 		pr_err("Error key code %#x out of range!\n", keyboard_send_msg.cmd);
		// 		return -1;
		// 	}
		// }
		if(cmd == KEYBOARD_IOC_KEY_LED_ON) {
			keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_KEY_LED_ON;
			set_bit(KEY_LED_ON_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		}
		else {
			keyboard_send_msg.cmd_type = FRAME_CMD_TYPE_KEY_LED_OFF;
			set_bit(KEY_LED_OFF_RESPONSE_BIT, &wait_respon_bits);   //置高某一位
		}
		break;
	}
	default:
		pr_err("Error non-supported cmd %d\n", cmd);
		return -7;
	}
	keyboard_send_msg.cmd_verify = FRAME_VERIFY(
			keyboard_send_msg.cmd_header0,
			keyboard_send_msg.cmd_header1,
			keyboard_send_msg.cmd_type,
			keyboard_send_msg.cmd,
			0,
			0);
#if 1
	if(debug_print)
		pr_err("i2c_master_send: h0 = %#x, h1 = %#x type = %#x, %#x, %#x\n",
			keyboard_send_msg.cmd_header0,
			keyboard_send_msg.cmd_header1,
			keyboard_send_msg.cmd_type,
			keyboard_send_msg.cmd,
			keyboard_send_msg.cmd_verify);
#endif
	mutex_lock(&iic_idel_Mutex);   //不让发
	mutex_lock(&iic_Mutex);  //加锁
	if(i2c_master_send(s_jc_keyboard_info.i2c_client, (char *)&keyboard_send_msg, sizeof(KEYBOARD_I2C_SEND_MSG_S)) != sizeof(KEYBOARD_I2C_SEND_MSG_S)) {
		pr_err("Error i2c_master_send!\n");
		mutex_unlock(&iic_Mutex);  //开锁
		mutex_unlock(&iic_idel_Mutex);  //开锁
		return -8;
	}
	mutex_unlock(&iic_Mutex);//开锁
	if(down_timeout(&s_jc_keyboard_info.ioctl_sem, JC_KEYBOARD_I2C_TIMEOUT/4)) {
		//没有中断的情况会超时
#ifndef NO_POLL_MODE
		if(s_jc_keyboard_inttimeout_func_t())	
#endif
		{
			pr_err("Error down_timeout!\n");
			mutex_unlock(&iic_idel_Mutex);  //开锁
			return -9;
		}
	}
	mutex_unlock(&iic_idel_Mutex);  //开锁



	switch(cmd) {
	case KEYBOARD_IOC_GET_BRIGHTNESS:
		if(test_bit(SET_BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_GET_BRIGHTNESS down_timeout! ----test_bit\n");
			clear_bit(SET_BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits);
			return -20;
		}
		if(!argv || copy_to_user((void *)argv, &s_i2c_reply_ret, 1)) {
			pr_err("Error copy_to_user!\n");
			return -10;
		}
		break;
	case KEYBOARD_IOC_GET_PANEL_MODEL:
		if(test_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_GET_PANEL_MODEL down_timeout! ----test_bit\n");
			clear_bit(PANEL_MODEL_RESPONSE_BIT, &wait_respon_bits);
			return -21;
		}
		if(!argv || copy_to_user((void *)argv, &s_i2c_reply_ret, 1)) {
			pr_err("Error copy_to_user!\n");
			return -10;
		}
		break;
	case KEYBOARD_IOC_GET_PANEL_VER:
		if(test_bit(PANEL_VER_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_GET_PANEL_VER down_timeout! ----test_bit\n");
			clear_bit(PANEL_VER_RESPONSE_BIT, &wait_respon_bits);
			return -22;
		}
		if(!argv || copy_to_user((void *)argv, &s_i2c_reply_ret, 1)) {
			pr_err("Error copy_to_user!\n");
			return -10;
		}
		break;
	case KEYBOARD_IOC_SET_BRIGHTNESS:
		if(test_bit(SET_BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_SET_BRIGHTNESS down_timeout! ----test_bit\n");
			clear_bit(SET_BRIGHTNESS_RESPONSE_BIT, &wait_respon_bits);
			return -23;
		}
		if(s_i2c_reply_ret == FRAME_CMD_REPLY_FAILED) {
			pr_err("Error execute failed!\n");
			return -11;
		}
		break;
	case KEYBOARD_IOC_RESET:
		if(test_bit(RESET_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_RESET down_timeout! ----test_bit\n");
			clear_bit(RESET_RESPONSE_BIT, &wait_respon_bits);
			return -24;
		}
		if(s_i2c_reply_ret == FRAME_CMD_REPLY_FAILED) {
			pr_err("Error execute failed!\n");
			return -11;
		}
		break;	
	case KEYBOARD_IOC_KEY_LED_ON:
		if(test_bit(KEY_LED_ON_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_KEY_LED_ON down_timeout! ----test_bit\n");
			clear_bit(KEY_LED_ON_RESPONSE_BIT, &wait_respon_bits);
			return -25;
		}
		if(s_i2c_reply_ret == FRAME_CMD_REPLY_FAILED) {
			pr_err("Error execute failed!\n");
			return -11;
		}
		break;
	case KEYBOARD_IOC_KEY_LED_OFF:   //灯的控制部分,不再等待应答,2022-09-05		
		if(test_bit(KEY_LED_OFF_RESPONSE_BIT, &wait_respon_bits))
		{
			pr_err("Error KEYBOARD_IOC_KEY_LED_OFF down_timeout! ----test_bit\n");
			clear_bit(KEY_LED_OFF_RESPONSE_BIT, &wait_respon_bits);
			return -26;
		}//return 0;
		if(s_i2c_reply_ret == FRAME_CMD_REPLY_FAILED) {
			pr_err("Error execute failed!\n");
			return -11;
		}
		break;
	default:
		if((cmd & 0xfc) == KEYBOARD_IOC_KEY_LED_FLASH)  //闪烁的指令
		{
			if(test_bit(KEY_LED_FLASH_RESPONSE_BIT, &wait_respon_bits))
			{
				pr_err("Error KEYBOARD_IOC_KEY_LED_FLASH down_timeout! ----test_bit\n");
				clear_bit(KEY_LED_FLASH_RESPONSE_BIT, &wait_respon_bits);
				return -27;
			}
			return 0;		
		}
		break;
	}
	return 0;
}

static struct file_operations s_jc_keyboard_fops = {
		.unlocked_ioctl = s_jc_keyboard_unlocked_ioctl,
};

static int s_jc_keyboard_misc_init(void) {
	memset(&s_jc_keyboard_miscdevice, 0, sizeof(struct miscdevice));
	s_jc_keyboard_miscdevice.minor = JC_KEYBOARD_MISCDEV_MINOR;
	s_jc_keyboard_miscdevice.name = JC_KEYBOARD_DRIVER_NAME;
	s_jc_keyboard_miscdevice.fops = &s_jc_keyboard_fops;
	if(misc_register(&s_jc_keyboard_miscdevice)) {
		pr_err("Error misc_register!\n");
		return -1;
	}
	sema_init(&s_jc_keyboard_info.ioctl_sem, 0);
	return 0;
}

static int s_jc_keyboard_misc_exit(void) {

	misc_deregister(&s_jc_keyboard_miscdevice);
	return 0;
}


static int jc_kayboard_command_for_read(struct i2c_client *client)
{
	int ret = -1;
	uint8_t tmpbuf[10] = {0};
	struct i2c_msg msg[1];

	msg[0].addr = client->addr;
	msg[0].flags = I2C_M_RD;/*Read*/
	msg[0].len = 7;
	msg[0].buf = tmpbuf;
	ret = i2c_transfer(client->adapter, msg, 1);
	return ret;
}


//static void s_jc_keyboard_exit(void);

static int s_jc_keyboard_probe(struct i2c_client *i2c_client, const struct i2c_device_id *i2c_device_id) {
	
	int ret;
//	jc_keyboard_inited = -1;
	if(debug_print)
		pr_err("addr = %#x\n", i2c_client->addr);

	ret = jc_kayboard_command_for_read(i2c_client);  //尝试着读一次
	if(ret < 0)  //读不了直接返回
	{
		pr_err("error : jc_keyboard_command_for_read,no probe\n");
//		s_jc_keyboard_exit();
		return ret;
	}


	memset(&s_jc_keyboard_info, 0, sizeof(JC_KEYBOARD_INFO));
	if((s_jc_keyboard_info.irq_gpio = of_get_named_gpio(i2c_client->dev.of_node, "irq-gpio", 0)) < 0) {
		pr_err("Error of_get_named_gpio %s\n", "irq-gpio");
		return -1;
	}
	if((i2c_client->irq = gpio_to_irq(s_jc_keyboard_info.irq_gpio)) < 0) {
		pr_err("Error gpio_to_irq %d\n", s_jc_keyboard_info.irq_gpio);
		return -1;
	}
	s_jc_keyboard_info.i2c_client = i2c_client;

	if(s_jc_keyboard_input_init()) {
		pr_err("Error s_jc_keyboard_input_init!\n");
		return -1;
	}
	if(s_jc_keyboard_misc_init()) {
		pr_err("Error s_jc_keyboard_misc_init!\n");
		s_jc_keyboard_input_exit();
		return -1;
	}
	if(s_jc_keyboard_irq_init()) {
		pr_err("Error s_jc_keyboard_irq_init!\n");
		s_jc_keyboard_misc_exit();
		s_jc_keyboard_input_exit();
		return -1;
	}
	
	 mutex_init(&iic_Mutex);  //初始化互斥锁
	 mutex_init(&iic_idel_Mutex);
//	jc_keyboard_inited = 1;  //加载成功!!
	return 0;
}

static int s_jc_keyboard_remove(struct i2c_client *i2c_client) {

	// if(jc_keyboard_inited != 1)  //驱动没有加载成功!!
	// 	return -1;

	if(s_jc_keyboard_irq_exit()) {
		pr_err("Error s_jc_keyboard_irq_exit!\n");
	}
	if(s_jc_keyboard_misc_exit()) {
		pr_err("Error s_jc_keyboard_misc_exit!\n");
	}
	if(s_jc_keyboard_input_exit()) {
		pr_err("Error s_jc_keyboard_input_exit!\n");
	}
	return 0;
}

static struct i2c_driver s_i2c_driver_jc_keyboard = {
		.probe = s_jc_keyboard_probe,
		.remove = s_jc_keyboard_remove,
		.driver = {
				.name = JC_KEYBOARD_DRIVER_NAME,
				.owner = THIS_MODULE,
				.of_match_table = s_jc_keyboard_match_table,
		},
		.id_table = s_jc_keyboard_i2c_id,
};

// static int s_jc_keyboard_init(void) {
// 	if(i2c_add_driver(&s_i2c_driver_jc_keyboard)) {
// 		pr_err("Error s_jc_keyboard_init i2c_add_driver!\n");
// 		return -1;
// 	}
// 	return 0;
// }

// static void s_jc_keyboard_exit(void) {
// 	return;
// //	i2c_del_driver(&s_i2c_driver_jc_keyboard);
// }


module_i2c_driver(s_i2c_driver_jc_keyboard);     //2023-03-30  增加 

// module_init(s_jc_keyboard_init);
// module_exit(s_jc_keyboard_exit);
// date 需要修改内核的makefile ,注释 886 KBUILD_CFLAGS   += $(call cc-option,-Werror=date-time)
MODULE_DESCRIPTION("Buildtime :"__DATE__" "__TIME__);
MODULE_AUTHOR("dazhi@jc,keyboard,2023-05");
MODULE_LICENSE("GPL");
MODULE_VERSION("1.1.0");    //2023-05-10 版本1.1.0

这里做了一个简单的分析,解决iic通信时的一个互斥问题,但是可能不是很完善,请各位多多包含指教。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/751715.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

新手杯—easy_base

0x00 前言 CTF 加解密合集&#xff1a;CTF 加解密合集 0x01 题目 0XezFWZfNXafRjNlNXYit3dvh2cmR3Y0x02 Write Up 先倒序 然后base64解码 以上

Self-Attention Cross-Attention

transformer的细节到底是怎么样的&#xff1f;Transformer 连环18问&#xff01; 4.1 从功能角度&#xff0c;Transformer Encoder的核心作用是提取特征&#xff0c;也有使用Transformer Decoder来提取特征。例如&#xff0c;一个人学习跳舞&#xff0c;Encoder是看别人是如何…

智能网卡在分布式 SDN 网络的应用与实践 | 龙蜥技术

编者按&#xff1a;当前智能网卡能够加速数据处理和传输&#xff0c;并能实现网络、存储和安全等功能卸载&#xff0c;在云计算领域得到广泛的应用。今天&#xff0c;浪潮数据云计算网络架构师王培辉带大家了解智能网卡加速原理和以及在浪潮分布式 SDN 网络加速的应用&#xff…

我连夜咨询了30个老同学,学IT上培训班到底有用么?

文章目录 一、背景二、学习IT上培训班的益处2.1 IT行业本身还不错2.2 获取到系统的专业知识2.3 获取到实战经验2.4 获取到网络资源和支持2.5 获取到职业发展指导2.6 建立初步的职业圈子人脉 三、学习IT上培训班的风险3.1 质量风险3.2 课程更新速度风险3.2 缺乏互动与实践机会风…

积分微分电路

积分微分电路 通过写出时域的推导&#xff0c;再到频域&#xff0c;详细介绍了积分微分的频率响应的推导&#xff0c;手绘了bode图&#xff0c;并仿真电路得到对应的结果。积分的频率响应&#xff1a;频率增加10倍&#xff0c;增益下降20db。输出相位超前输入相位90度。微分的…

GPT-4 最强竞争对手,Claude 杀疯了!

公众号关注 “GitHubDaily” 设为 “星标”&#xff0c;每天带你逛 GitHub&#xff01; 在今年早些时候&#xff0c;ChatGPT、Bard、Claude 等大语言模型&#xff0c;在 AI 领域呈三权鼎立之势&#xff0c;无人能出其右&#xff0c;被视为是能力表现最为卓越的 3 款 AI 聊天机器…

阿里云无影云电脑具体价格_云桌面不同配置1元报价

阿里云无影云电脑配置费用&#xff0c;4核8G企业办公型云电脑可以免费使用3个月&#xff0c;无影云电脑地域不同费用不同&#xff0c;无影云电脑是由云桌面配置、云盘、互联网访问带宽、AD Connector、桌面组共用桌面session等费用组成&#xff0c;阿里云百科分享阿里云无影云电…

大模型的“第一性原理”:技术创新与社会价值的接轨

随着时间来到2023年第三季度&#xff0c;国产大模型已经达到100多个&#xff0c;“百模大战”正式开启。 大模型&#xff0c;我们有了很多选择&#xff0c;也开始呈现出某种同质化。除了拼参数、比背景、看榜单&#xff0c;有没有其他方法&#xff0c;让我们更好地判断一个大模…

解决Gson解析json字符串,Integer变为Double类型的问题

直接上代码记录下。我代码里没有Gson包&#xff0c;用的是nacos对Gson的封装&#xff0c;只是包不同&#xff0c;方法都一样 import com.alibaba.nacos.shaded.com.google.common.reflect.TypeToken; import com.alibaba.nacos.shaded.com.google.gson.*;import java.util.Map;…

经典CNN(一):ResNet-50算法实战与解析

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊|接辅导、项目定制 1 ResNet理论 深度残差网络ResNet(deep residual network)在2015年由何凯明等提出&#xff0c;因为它简单与实用并存&#xff0c;随后很多研究…

Hutool工具类 -集常用工具类为一体 - 工具类之大成

文章目录 说在前面的话简介gitee介绍项目介绍 网址gtiee 网址github 网址 安装pom依赖引入 &#xff1a;下载jar 文档中文文档中文备用文档参考API视频介绍 部分截图首页包含组件(总)IO流相关部分工具类(Util)集合类HTTP客户端 功能不再一一赘述和截图&#xff0c;具体请查看官…

详解TCP协议

TCP协议段格式 序号和确认序号&#xff1a;在真实服务器和客服端通信过程中请求是并行执行的&#xff0c;这会导致到达是乱序的&#xff0c;所以才会有序号这个东西&#xff0c;确认序号是对方应答时返回的&#xff0c;例如序号发送到1&#xff0c;确认序号会返回2&#xff0c;…

计算机网络 day6 arp病毒 - ICMP协议 - ping命令 - Linux手工配置IP地址

目录 arp协议 arp病毒\欺骗 arp病毒的运行原理 arp病毒产生的后果&#xff1a; 解决方法&#xff1a; ICMP协议 ICMP用在哪里&#xff1f; ICMP协议数据的封装过程 ​编辑 为什么icmp协议封装好数据后&#xff0c;还要加一个ip包头&#xff0c;再使用ip协议再次进…

springboot农机电招平台

本系统为了数据库结构的灵活性所以打算采用MySQL来设计数据库&#xff0c;而java技术&#xff0c;B/S架构则保证了较高的平台适应性。本文主要介绍了本系统的开发背景&#xff0c;所要完成的功能和开发的过程&#xff0c;主要说明了系统设计的重点、设计思想。 本系统主要是设…

关于java垃圾回收的小结

一、为什么要有垃圾回收 我们每次创建对象都需要在栈上开辟空间&#xff0c;堆上使用内存&#xff0c;如果我们只是开辟了这个空间&#xff0c;而不去释放他&#xff0c;那么再大的内存和空间也会有满的一天&#xff0c;所以我们在Java中引入了GC&#xff08;垃圾回收机制&…

Foxit PDF ActiveX 5.9.8 Crack

Foxit PDF SDK ActiveX 即时添加PDF显示功能至Windows应用程序&#xff0c;快速投放市场&#xff0c;可视化编程组件功能强大且易于使用的PDF软件开发工具包 对于刚接触PDF或不愿投入过多精力学习PDF技术的产品管理者及开发者来说&#xff0c;Foxit PDF SDK ActiveX无疑是理想…

中国1km分辨率逐月平均气温数据集(1901-2022)

时间分辨率月空间分辨率1km - 10km共享方式开放获取数据大小9.71 GB数据时间范围 1901.1-2022.12 数据集摘要 该数据为中国逐月平均温度数据,空间分辨率为0.0083333(约1km),时间为1901.1-2022.12。数据格式为NETCDF,即.nc格式。数据单位为0.1 ℃。该数据集是根据CRU发布的…

对Vue组件化开发思想的一些理解

目录 组件的分类 为什么需要组件化开发 如何设计组件 组件间通信 组件系统是 Vue的一个重要概念&#xff0c;让我们可以用独立可复用的小组件来构建大型应用。几乎任意类型的应用的界面都可以抽象为一个组件树&#xff1a; 写一个 Vue 项目&#xff0c;其实就是在写一个个的…

接口测试 react+unittest+flask 接口自动化测试平台

目录 1 前言 2 框架 2-1 框架简介 2-2 框架介绍 2-3 框架结构 3 平台 3-1 平台组件图 1 新建用例 2 生成测试任务 3 执行并查看测试报告 3-2 用例管理 3-2-1 用例设计 3-3 任务管理 3-3-1 创建任务 3-3-2 执行任务 3-3-3 测试报告 3-3-4 邮件通知 1 前言 构建…

idea新建xml模板设置,例如:mybatis-config

在idea怎么新建mapper.xml文件&#xff0c;具体操作步骤和结果如下&#xff0c;其他文件也是可以自定义模板的流程和步骤一致&#xff01; 效果如下&#xff1a; 步骤如图&#xff1a; step1&#xff1a; step2&#xff1a; 文件内容&#xff1a; <?xml version"…