linux的socket CAN驱动介绍

2023-05-16

https://blog.csdn.net/linyangspring/article/details/27186911

在linux中,CAN总线的驱动有两种实现方式:字符设备以及socket can驱动。Socket CAN使用伯克利的Socket接口和Linux网络协议栈,这种方法使得CAN设备驱动可以通过网络接口来调用。Socket CAN的接口被设计的尽量接近TCP/IP的协议,让那些熟悉网络编程的程序员能够比较容易的学习和使用。

本文以赛灵思的Zynq-7000为硬件背景,详细介绍开发板上的socket can驱动。主要的驱动文件为dev.c以及xilinx_can.c,可以从https://github.com/Xilinx/linux-xlnx获取。

首先看一下传递给内核的dts文件中的can设备信息:

ps7_can_0: ps7-can@e0008000 {
	clock-names = "ref_clk", "aper_clk";
	clocks = <&clkc 19>, <&clkc 36>;
	compatible = "xlnx,ps7-can-1.00.a", "xlnx,ps7-can";
	interrupt-parent = <&ps7_scugic_0>;
	interrupts = <0 28 4>;
	reg = <0xe0008000 0x1000>;
} ;

指定了寄存器地址范围,中断号,驱动适配版本以及参考时钟源。

再来看xilinx_can.c文件中的代码:

/* Match table for OF platform binding */
static struct of_device_id xcan_of_match[] = {
	{ .compatible = "xlnx,ps7-can", },
	{ .compatible = "xlnx,axi-can-1.00.a", },
	{ /* end of list */ },
};
MODULE_DEVICE_TABLE(of, xcan_of_match);
 
static struct platform_driver xcan_driver = {
	.probe = xcan_probe,
	.remove	= xcan_remove,
	.driver	= {
		.owner = THIS_MODULE,
		.name = DRIVER_NAME,
		.pm = &xcan_dev_pm_ops,
		.of_match_table	= xcan_of_match,
	},
};

可见对于PS的CAN接口以及PL的基于axi总线的CAN接口,均可以使用该platform_driver驱动。

ARM上电之后,linux初始化函数会依据dts信息将CAN硬件信息加入到系统的硬件链表中,当驱动程序装载时,会去遍历该链表获取硬件信息,比如寄存器地址、中断号等,然后调用ioremap、request_irq等,进一步初始化硬件。

接着看xcan_probe函数:

/**
 * xcan_probe - Platform registration call
 * @pdev:	Handle to the platform device structure
 *
 * This function does all the memory allocation and registration for the CAN
 * device.
 *
 * Return: 0 on success and failure value on error
 */
static int xcan_probe(struct platform_device *pdev)
{
	struct resource *res; /* IO mem resources */
	struct net_device *ndev;
	struct xcan_priv *priv;
	struct device *dev = &pdev->dev;
	int ret, irq;
 
	/* Create a CAN device instance */
	ndev = alloc_candev(sizeof(struct xcan_priv), XCAN_ECHO_SKB_MAX);
	if (!ndev)
		return -ENOMEM;
 
	priv = netdev_priv(ndev);
	priv->dev = ndev;
	priv->can.bittiming_const = &xcan_bittiming_const;
	priv->can.do_set_bittiming = xcan_set_bittiming;
	priv->can.do_set_mode = xcan_do_set_mode;
	priv->can.do_get_berr_counter = xcan_get_berr_counter;
	priv->can.ctrlmode_supported = CAN_CTRLMODE_LOOPBACK;
	priv->waiting_ech_skb_index = 0;
	priv->ech_skb_next = 0;
	priv->waiting_ech_skb_num = 0;
	priv->xcan_echo_skb_max = XCAN_ECHO_SKB_MAX;
 
	/* Get IRQ for the device */
	ndev->irq = platform_get_irq(pdev, 0);
	irq = devm_request_irq(&pdev->dev, ndev->irq, &xcan_interrupt,
				priv->irq_flags, dev_name(&pdev->dev),
				(void *)ndev);
	if (irq < 0) {
		ret = irq;
		dev_err(&pdev->dev, "Irq allocation for CAN failed\n");
		goto err_free;
	}
 
	spin_lock_init(&priv->ech_skb_lock);
	ndev->flags |= IFF_ECHO;	/* We support local echo */
 
	platform_set_drvdata(pdev, ndev);
	SET_NETDEV_DEV(ndev, &pdev->dev);
	ndev->netdev_ops = &xcan_netdev_ops;
 
	/* Get the virtual base address for the device */
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	priv->reg_base = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(priv->reg_base)) {
		ret = PTR_ERR(priv->reg_base);
		goto err_free;
	}
	ndev->mem_start = res->start;
	ndev->mem_end = res->end;
 
	priv->write_reg	= xcan_write_reg;
	priv->read_reg = xcan_read_reg;
 
	/* Getting the CAN devclk info */
	priv->devclk = devm_clk_get(&pdev->dev, "ref_clk");
	if (IS_ERR(priv->devclk)) {
		dev_err(&pdev->dev, "Device clock not found.\n");
		ret = PTR_ERR(priv->devclk);
		goto err_free;
	}
 
	/* Check for type of CAN device */
	if (of_device_is_compatible(pdev->dev.of_node, "xlnx,ps7-can")) {
		priv->aperclk = devm_clk_get(&pdev->dev, "aper_clk");
		if (IS_ERR(priv->aperclk)) {
			dev_err(&pdev->dev, "aper clock not found\n");
			ret = PTR_ERR(priv->aperclk);
			goto err_free;
		}
	} else {
		priv->aperclk = priv->devclk;
	}
 
	ret = clk_prepare_enable(priv->devclk);
	if (ret) {
		dev_err(&pdev->dev, "unable to enable device clock\n");
		goto err_free;
	}
 
	ret = clk_prepare_enable(priv->aperclk);
	if (ret) {
		dev_err(&pdev->dev, "unable to enable aper clock\n");
		goto err_unprepar_disabledev;
	}
 
	priv->can.clock.freq = clk_get_rate(priv->devclk);
 
	ret = register_candev(ndev);
	if (ret) {
		dev_err(&pdev->dev, "fail to register failed (err=%d)\n", ret);
		goto err_unprepar_disableaper;
	}
 
	dev_info(&pdev->dev,
			"reg_base=0x%p irq=%d clock=%d, tx fifo depth:%d\n",
			priv->reg_base, ndev->irq, priv->can.clock.freq,
			priv->xcan_echo_skb_max);
 
	return 0;
 
err_unprepar_disableaper:
	clk_disable_unprepare(priv->aperclk);
err_unprepar_disabledev:
	clk_disable_unprepare(priv->devclk);
err_free:
	free_candev(ndev);
 
	return ret;
}

当一个设备驱动通过driver_register加入对应的驱动总线下时,会去遍历对应总线下的设备双向链表,当驱动和设备匹配时,会触发驱动的probe函数,probe函数的传入参数pdev即为遍历得到的设备信息。

struct xcan_priv为CAN私有数据结构,包含struct can_priv、struct net_device等数据成员,注意can_priv结构成员一定要放在第一位,具体原因参考http://blog.sina.com.cn/s/blog_636a55070101mc2d.html。

/**
 * struct xcan_priv - This definition define CAN driver instance
 * @can:			CAN private data structure.
 * @open_time:			For holding timeout values
 * @waiting_ech_skb_index:	Pointer for skb
 * @ech_skb_next:		This tell the next packet in the queue
 * @waiting_ech_skb_num:	Gives the number of packets waiting
 * @xcan_echo_skb_max:		Maximum number packets the driver CAN send
 * @ech_skb_lock:		For spinlock purpose
 * @read_reg:			For reading data from CAN registers
 * @write_reg:			For writing data to CAN registers
 * @dev:			Network device data structure
 * @reg_base:			Ioremapped address to registers
 * @irq_flags:			For request_irq()
 * @aperclk:			Pointer to struct clk
 * @devclk:			Pointer to struct clk
 */
struct xcan_priv {
	struct can_priv can;
	int open_time;
	int waiting_ech_skb_index;
	int ech_skb_next;
	int waiting_ech_skb_num;
	int xcan_echo_skb_max;
	spinlock_t ech_skb_lock;
	u32 (*read_reg)(const struct xcan_priv *priv, int reg);
	void (*write_reg)(const struct xcan_priv *priv, int reg, u32 val);
	struct net_device *dev;
	void __iomem *reg_base;
	unsigned long irq_flags;
	struct clk *aperclk;
	struct clk *devclk;
};

调用alloc_candev()函数获取一个net_device变量,设置socket buffer大小为XCAN_ECHO_SKB_MAX=64个字节。接着是对struct xcan_priv *priv指针的初始化。

	priv->can.do_set_bittiming = xcan_set_bittiming;
	priv->can.do_set_mode = xcan_do_set_mode;
	priv->can.do_get_berr_counter = xcan_get_berr_counter;

CAN接口的位速率设置函数,模式设置函数,数据传输错误计数函数。

牵涉到CAN接口的具体操作函数代码如下:

static const struct net_device_ops xcan_netdev_ops = {
	.ndo_open	= xcan_open,
	.ndo_stop	= xcan_close,
	.ndo_start_xmit	= xcan_start_xmit,
};
ndev->netdev_ops = &xcan_netdev_ops;

主要是指定struct net_device *ndev指针的打开关闭以及数据发送函数。

然后指定struct xcan_priv *priv指针的读写寄存器函数。

static void xcan_write_reg(const struct xcan_priv *priv, int reg, u32 val)
{
	writel(val, priv->reg_base + reg);
}
static u32 xcan_read_reg(const struct xcan_priv *priv, int reg)
{
	return readl(priv->reg_base + reg);
}
.............
 
        priv->write_reg = xcan_write_reg;
	priv->read_reg = xcan_read_reg;

接着获取设备时钟源并使能,注意Zynq-7000要求每个设备有两个时钟源。

然后注册CAN设备,其实是注册net设备,只不过指定net设备的操作函数为CAN特定的操作函数。

ret = register_candev(ndev);
............
int register_candev(struct net_device *dev)
{
	dev->rtnl_link_ops = &can_link_ops;
	return register_netdev(dev);
}
.......
static struct rtnl_link_ops can_link_ops __read_mostly = {
	.kind		= "can",
	.maxtype	= IFLA_CAN_MAX,
	.policy		= can_policy,
	.setup		= can_setup,
	.newlink	= can_newlink,
	.changelink	= can_changelink,
	.get_size	= can_get_size,
	.fill_info	= can_fill_info,
	.get_xstats_size = can_get_xstats_size,
	.fill_xstats	= can_fill_xstats,
};

至于xcan_remove()函数不再详述。

static int xcan_remove(struct platform_device *pdev)
{
	struct net_device *ndev = platform_get_drvdata(pdev);
	struct xcan_priv *priv = netdev_priv(ndev);
 
	if (set_reset_mode(ndev) < 0)
		netdev_err(ndev, "mode resetting failed!\n");
 
	unregister_candev(ndev);
	clk_disable_unprepare(priv->aperclk);
	clk_disable_unprepare(priv->devclk);
 
	free_candev(ndev);
 
	return 0;
}

有关socket can应用层程序的编写可以参考Documentation\networking\can.txt。

当使用socket打开can接口时,会调用到xcan_open()函数:

static int xcan_open(struct net_device *ndev)
{
	struct xcan_priv *priv = netdev_priv(ndev);
	int err;
 
	/* Set chip into reset mode */
	err = set_reset_mode(ndev);
	if (err < 0)
		netdev_err(ndev, "mode resetting failed failed!\n");
 
	/* Common open */
	err = open_candev(ndev);
	if (err)
		return err;
 
	err = xcan_start(ndev);
	if (err < 0)
		netdev_err(ndev, "xcan_start failed!\n");
 
	priv->open_time = jiffies;
 
	can_led_event(ndev, CAN_LED_EVENT_OPEN);
	netif_start_queue(ndev);
 
	return 0;
}

首先reset CAN,进入config mode;然后调用dev.c中的open_candev()函数打开CAN接口;调用xcan_start()函数,进入Normal mode,主要是使能中断,依据应用层传入的mode参数设置loopback mode或者normal mode;然后使能CAN接口,并等待XCAN_SR_OFFSET寄存器进入对应的模式。

static int set_normal_mode(struct net_device *ndev)
{
	struct xcan_priv *priv = netdev_priv(ndev);
 
	/* Enable interrupts */
	priv->write_reg(priv, XCAN_IER_OFFSET, XCAN_IXR_TXOK_MASK |
			XCAN_IXR_BSOFF_MASK | XCAN_IXR_WKUP_MASK |
			XCAN_IXR_SLP_MASK | XCAN_IXR_RXNEMP_MASK |
			XCAN_IXR_ERROR_MASK | XCAN_IXR_ARBLST_MASK);
 
	/* Check whether it is loopback mode or normal mode  */
	if (priv->can.ctrlmode & CAN_CTRLMODE_LOOPBACK)
		/* Put device into loopback mode */
		priv->write_reg(priv, XCAN_MSR_OFFSET, XCAN_MSR_LBACK_MASK);
	else
		/* The device is in normal mode */
		priv->write_reg(priv, XCAN_MSR_OFFSET, 0);
 
	if (priv->can.state == CAN_STATE_STOPPED) {
		/* Enable Xilinx CAN */
		priv->write_reg(priv, XCAN_SRR_OFFSET, XCAN_SRR_CEN_MASK);
		priv->can.state = CAN_STATE_ERROR_ACTIVE;
		if (priv->can.ctrlmode & CAN_CTRLMODE_LOOPBACK) {
			while ((priv->read_reg(priv, XCAN_SR_OFFSET) &
					XCAN_SR_LBACK_MASK) == 0)
					;
		} else {
			while ((priv->read_reg(priv, XCAN_SR_OFFSET)
					& XCAN_SR_NORMAL_MASK) == 0)
					;
		}
		netdev_dbg(ndev, "status:#x%08x\n",
				priv->read_reg(priv, XCAN_SR_OFFSET));
	}
 
	return 0;
}

最后调用netif_start_queue()函数使能发送队列。

对应的xcan_close()函数不再分析。

static int xcan_close(struct net_device *ndev)
{
	struct xcan_priv *priv = netdev_priv(ndev);
 
	netif_stop_queue(ndev);
	if (set_reset_mode(ndev) < 0)
		netdev_err(ndev, "mode resetting failed failed!\n");
 
	close_candev(ndev);
 
	priv->open_time = 0;
 
	can_led_event(ndev, CAN_LED_EVENT_STOP);
 
	return 0;
}

带NAPI的中断轮询数据接收模式

中断接收数据模式在数据频繁情况下,中断触发负载过大,系统性能受到影响,为此基于轮询的接收模式被开发,称为New API,即NAPI。

NAPI仍然需要首次数据包接收中断来触发poll过程,第一次接收中断发生后,中断处理程序禁止设备的接收中断,通过poll方式读取设备的接收缓冲区后,再次使能中断。

NAPI函数的调用过程如下:

netif_napi_add

napi_enable

关中断

napi_schedule

netif_receive_skb

napi_complete

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

linux的socket CAN驱动介绍 的相关文章

  • 如何为 Linux 桌面条目文件指定带有相对路径的图标?

    对于我的一个 Linux 应用程序 我有应用程序二进制文件 一个 launcher sh 脚本 针对 LD LIBRARY PATH 和一个 desktop 文件 所有这些都位于同一文件夹中 我想使用图标的相对路径而不是绝对路径 我试过了
  • 我们真的应该使用 Chef 来管理 sudoers 文件吗?

    这是我的问题 我担心如果 Chef 破坏了 sudoers 文件中的某些内容 可能是 Chef 用户错误地使用了说明书 那么服务器将完全无法访问 我讨厌我们完全失去客户的生产服务器 因为我们弄乱了 sudoers 文件并且无法再通过 ssh
  • 在 Linux 上以编程方式设置 DNS 名称服务器

    我希望能够通过我的 C C 程序为 Linux 上的 DNS 名称服务器添加 IP 地址 我在一个带有只读 etc resolv conf 的嵌入式平台上 这意味着我不能简单地将 nameserver xxx xxx xxx xxx 行添加
  • Linux 上的静态 Qt5 构建:部署时如何处理字体?

    我使用这些配置选项创建了 Qt 5 2 0 库的静态版本 Ubuntu 12 04 开源 确认许可 force pkg config 发布 静止的 前缀 home juzzlin qt5 无icu opengl桌面 无油嘴滑舌 辅助功能 n
  • 如何阻止ubuntu在使用apt安装或更新软件包时弹出“Daemons using outdatedlibraries”? [关闭]

    Closed 这个问题是与编程或软件开发无关 help closed questions 目前不接受答案 我最近新安装了 Ubuntu 22 04 LTS 我发现每次使用 apt 安装或更新软件包时 它都会询问我有关Which servic
  • GMail 421 4.7.0 稍后重试,关闭连接

    我试图找出为什么它无法使用 GMail 从我的服务器发送邮件 为此 我使用 SwiftMailer 但我可以将问题包含在以下独立代码中
  • C 语言的符号表

    我目前正在开发一种执行模式匹配的静态分析工具 我在用Flex https github com westes flex生成词法分析器 我编写了代码来管理符号表 我不太有经验C 所以我决定将符号表实现为线性链表 include
  • 添加文件时运行 shell 命令

    我的 Linux 机器上有一个名为 images 的文件夹 该文件夹连接到一个网站 该网站的管理员可以向该网站添加图片 但是 当添加图片时 我想要一个命令来运行调整目录中所有图片的大小 简而言之 我想知道当新文件添加到特定位置时如何使服务器
  • 并行运行 shell 脚本

    我有一个 shell 脚本 打乱大型文本文件 600 万行和 6 列 根据第一列对文件进行排序 输出 1000 个文件 所以伪代码看起来像这样 file1 sh bin bash for i in seq 1 1000 do Generat
  • 使用 shell 脚本将行附加到 /etc/hosts 文件

    我有一个新的 Ubuntu 12 04 VPS 我正在尝试编写一个安装脚本来完成整个 LAMP 安装 我遇到问题的地方是在 etc hosts文件 我当前的主机文件如下所示 127 0 0 1 localhost Venus The fol
  • 如何使用waf构建共享库?

    我想使用构建一个共享库waf http code google com p waf 因为它看起来比 GNU 自动工具更容易 更简洁 到目前为止 我实际上有几个与我开始编写的 wscript 有关的问题 VERSION 0 0 1 APPNA
  • 与 pthread 的进程间互斥

    我想使用一个互斥体 它将用于同步对两个不同进程共享的内存中驻留的某些变量的访问 我怎样才能做到这一点 执行该操作的代码示例将非常感激 以下示例演示了 Pthread 进程间互斥体的创建 使用和销毁 将示例推广到多个进程作为读者的练习 inc
  • 静态方法的 Java 内存模型

    我来自操作系统和 C 语言背景 在代码编译时 世界很简单 需要处理和理解堆栈 堆文本部分等 当我开始学习 Java 时 我确实了解 JVM 和垃圾收集器 我对静态方法感到很有趣 根据我的理解 类的所有实例都会在堆中创建 然后被清理 但是 对
  • 使用 python 脚本更改 shell 中的工作目录

    我想实现一个用户态命令 它将采用其参数之一 路径 并将目录更改为该目录 程序完成后 我希望 shell 位于该目录中 所以我想实施cd命令 但需要外部程序 可以在 python 脚本中完成还是我必须编写 bash 包装器 Example t
  • Linux 为一组进程保留一个处理器(动态)

    有没有办法将处理器排除在正常调度之外 也就是说 使用sched setaffinity我可以指示线程应该在哪个处理器上运行 但我正在寻找相反的情况 也就是说 我想从正常调度中排除给定的处理器 以便只有已明确调度的进程才能在那里运行 我还知道
  • cdc_acm:无法设置 dtr/rts - 无法与 USB cdc 设备通信

    我试图使用 pic24fj128gb206 枚举 usb cdc 设备 设备似乎已正确枚举 但是当我将设备连接到 Linux PC 时 我从内核收到以下警告消息 cdc acm 1 8 1 6 7 1 0 failed to set dtr
  • 复制目录内容

    我想将目录 tmp1 的内容复制到另一个目录 tmp2 tmp1 可能包含文件和其他目录 我想使用C C 复制tmp1的内容 包括模式 如果 tmp1 包含目录树 我想递归复制它们 最简单的解决方案是什么 我找到了一个解决方案来打开目录并读
  • 在运行时检查 GCC 版本

    我需要找出 C 程序执行过程中 运行时 可用的 系统中安装的 GCC 版本 主要版本和次要版本 意思是 以编程方式提取可用 gcc 的版本 就像我在 shell 中输入 gcc version 一样 但在 c 程序中 The GNUC an
  • 为 OpenWrt 编写和编译程序

    我有一个在 OpenWRT 下运行的具有 MIPS 架构的嵌入式设备 系统类型 MediaTek MT7628AN ver 1 eco 2机器 WRTnode2P 处理器 0CPU型号 MIPS 24KEc V5 5 我想通过我的电脑 ub
  • 跨 CPU 内核的 rdtsc 精度

    我从一个线程发送网络数据包 并在运行于不同 CPU 核心上的第二个线程上接收回复 我的流程测量每个数据包发送和接收之间的时间 类似于 ping 我使用 rdtsc 来获得高分辨率 低开销的计时 这是我的实现所需要的 所有测量结果看起来都很可

随机推荐

  • mysql以某个字段分组进行条件查询,如果为0则补0

    需求 xff1a 有一个version plan表 xff0c 其中有一个字段is urgent 且每个version plan与一个模块moduleId关联 xff0c 现在要查出is urgent为1的模块数和这个模块的总数 xff0c
  • 数据库系统第六章-关系数据理论复习

    候选码若关系中的一个属性或属性组的值能够唯一地标识一个元组 xff0c 且他的子集不能唯一的标识一个元组 xff0c 则称这个属性或属性组做候选码 函数依赖 范式 虽然STC xff08 S T C xff09 3NF xff0c 但它仍存
  • Spring Security登录验证过程详解

    前端 xff1a 在前端页面输入username和password 通过地址login访问验证 后台 xff1a 调用 AbstractAuthenticationProcessingFilter doFilter 方法 原因 xff1a
  • 关于STL的一些理解

    1 集合set 定义 xff1a set lt int gt s1 类型可选 se begin 返回指向第一个元素的迭代器 se clear 清除所有元素 常用 se count 返回某个值元素的个数 常用 xff0c 一般用来查这个元素在
  • ModuleNotFoundError: No module named ‘...’:报错解决方案

    1 module包没安装 这个 就是包的名字 xff0c 看问题定位 xff0c import源文件 xff0c 然后去site packages里找那个模块 xff0c 如果是空的没安装 xff0c 这时候可以在对应的环境里使用pip i
  • C语言----结构体,枚举,共用体

    1 xff09 结构体 span class token comment 例 xff1a 一个描述学生的结构体 span span class token keyword struct span span class token class
  • Nvidia jetson agx xavier can通讯失败,修改时钟源pllaon

    一 问题详情 在调试Xavier can的过程中 xff0c can通信始终有问题 xff0c 收发都会报错 后来发现时钟源导致波特率不匹配 xff0c 所以不能通信 二 修改时钟源 在虚拟机使用官方sdkmanager xff0c 下载安
  • linux下实现https访问

    http转https 适用于linux服务器 linux下nginx 43 ssl实现https访问 xff08 一 xff09 环境准备 xff08 二 xff09 下载nginx源码 xff0c 编译nginx并添加ssl模块 xff0
  • Nvidia Jetson nano 安装Archiconda、gpu版torch、踩坑记录

    Nvidia Jetson nano 安装Archiconda gpu版torch 踩坑记录 jetson nano 属于aarch64架构 xff0c 不同于一般的x86 64的linux系统架构 xff0c 而torch官网上面 htt
  • linux下的yum,vim,gcc,gdb

    我们在windows系统下我们能够很简单进行粘贴复制下载安装 xff0c 也能通过vs进行我们缩写代码的编译运行 xff0c 但是我们在linux下我们应该怎样去操作我们的这些命令呢 xff0c 这章我们就讲解一下我们linux下常用的一些
  • 【Keil】Keil5添加源程序和头文件

    xxx c就是源程序 xxx h就是头文件 源程序添加方法 双击文件夹 xff0c 例如图片上的Source xff0c 跳出弹窗 xff0c 选择需要添加的源程序即可 添加头文件的方法 1 首先点击图片红框处 xff0c 或是在文件夹te
  • C 标准库 string常用函数 笔记

    文章目录 64 TOC 文章目录 前言字符串长度strlen sizeof 字符串拼接strcat strncat 字符串拷贝strcpy strncpy memcpy 内存填充memset 字符串比较strcmp strncmp 字符串查
  • node.js中的http.request方法使用说明_node.js

    方法说明 xff1a 函数的功能室作为客户端向HTTP服务器发起请求 语法 xff1a 复制代码 代码如下 http get options callback 由于该方法属于http模块 xff0c 使用前需要引入http模块 xff08
  • ARM中SP、LR、PC三个寄存器介绍

    寄存器定义和用途 定义 寄存器是中央处理器内的组成部份 寄存器是有限存贮容量的高速存贮部件 xff0c 它们可用来暂存指令 数据和位址 用途 可将寄存器内的数据执行算术及逻辑运算 xff1b 存于寄存器内的地址可用来指向内存的某个位置 xf
  • Linux buffer/cache介绍

    free 命令 与 buffer cache 在 Linux 系统中 xff0c 我们经常用 free m命令来查看系统内存的使用状态 xff1a m 显示单位为MB free m 各个参数的说明 total 内存总数 used 已经使用的
  • 数字签名算法RSA

    RSA RSA数字签名算法源于RSA公钥密码算法的思想 xff0c 将RSA公钥密码算法按照数字签名的方式运用 RSA数字签名算法是迄今为止应用最为广泛的数字签名算法 RSA数字签名算法的实现如RSA加密算法一致 RSA数字签名算法主要可分
  • AXI接口协议详解-AXI总线、接口、协议

    转自 xff1a https cloud tencent com developer article 1695010 AXI接口协议详解 AXI总线 接口 协议 AXI 总线 上面介绍了AMBA总线中的两种 xff0c 下面看下我们的主角
  • USB主机是如何检测到设备的插入的呢?

    转自 xff1a https www cnblogs com wangh0802PositiveANDupward archive 2013 05 06 3061241 html USB设备的插入检测机制 首先 xff0c 在USB集线器的
  • 电路图中的那些类似于箭头的是什么意思?

    这个双箭头在画图软件中称为 off connect xff0c 作用是连接一个工程中的两张原理图里网络标号相同的网络 即 xff1a 这张原理图中网络标号为 PCIE20 2 REFCLKP 网络 xff0c 与其他原理图中的网络标号为 P
  • linux的socket CAN驱动介绍

    https blog csdn net linyangspring article details 27186911 在linux中 xff0c CAN总线的驱动有两种实现方式 xff1a 字符设备以及socket can驱动 Socket