NORFlash协议层

2023-12-20 07:18
文章标签 协议 norflash

本文主要是介绍NORFlash协议层,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、CFI mode:
①、注册 cfi_chipdrv :

static int __init cfi_probe_init(void)
{register_mtd_chip_driver(&cfi_chipdrv);return 0;
}
/* linux-3.4.2\drivers\mtd\chips\cfi_probe.c */

在cfi_probe_init函数中调用register_mtd_chip_driver函数,将cfi_chipdrv 添加到chip_drvs_list中去:

void register_mtd_chip_driver(struct mtd_chip_driver *drv)
{spin_lock(&chip_drvs_lock);list_add(&drv->list, &chip_drvs_list);spin_unlock(&chip_drvs_lock);
}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

其中,cfi_chipdrv 结构体变量定义如下:

static struct mtd_chip_driver cfi_chipdrv = {.probe      = cfi_probe,.name       = "cfi_probe",.module     = THIS_MODULE
};
/* linux-3.4.2\drivers\mtd\chips\cfi_probe.c */

②、do_map_probe执行过程:
在do_map_probe函数中调用get_mtd_chip_driver函数 :

struct mtd_info *do_map_probe(const char *name, struct map_info *map)
{struct mtd_chip_driver *drv;struct mtd_info *ret;drv = get_mtd_chip_driver(name);...}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

依次列举chip_drvs_list中的mtd_chip_driver 项,将每一项的成员name和传入参数name做比较,返回name相同的mtd_chip_driver 项,返回值ret = &cfi_chipdrv给do_map_probe函数中的drv:

static struct mtd_chip_driver *get_mtd_chip_driver (const char *name)
{struct list_head *pos;struct mtd_chip_driver *ret = NULL, *this;spin_lock(&chip_drvs_lock);list_for_each(pos, &chip_drvs_list) {this = list_entry(pos, typeof(*this), list);if (!strcmp(this->name, name)) {ret = this;break;}}if (ret && !try_module_get(ret->module))ret = NULL;spin_unlock(&chip_drvs_lock);return ret; //ret = &cfi_chipdrv
}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

再由mtd_chip_driver返回do_map_probe执行drv->probe(map):

struct mtd_info *do_map_probe(const char *name, struct map_info *map)
{struct mtd_chip_driver *drv;struct mtd_info *ret;...ret = drv->probe(map); //drv = &cfi_chipdrv...return ret;
}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

由于drv = &cfi_chipdrv,执行执行drv->probe即相当于执行cfi_chipdrv.probe = cfi_probe:

struct mtd_info *cfi_probe(struct map_info *map)
{/** Just use the generic probe stuff to call our CFI-specific* chip_probe routine in all the possible permutations, etc.*/return mtd_do_chip_probe(map, &cfi_chip_probe);
}
/* linux-3.4.2\drivers\mtd\chips\cfi_probe.c */

其中,cfi_chip_probe如下定义:

static struct chip_probe cfi_chip_probe = {.name       = "CFI",.probe_chip = cfi_probe_chip
};
/* linux-3.4.2\drivers\mtd\chips\cfi_probe.c */

cfi_probe ->mtd_do_chip_probe:

struct mtd_info *mtd_do_chip_probe(struct map_info *map, struct chip_probe *cp)
{struct mtd_info *mtd = NULL;struct cfi_private *cfi;/* First probe the map to see if we have CFI stuff there. */cfi = genprobe_ident_chips(map, cp); //cp = &cfi_chip_probe ...}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

mtd_do_chip_probe->genprobe_ident_chips:

static struct cfi_private *genprobe_ident_chips(struct map_info *map, struct chip_probe *cp)
{struct cfi_private cfi;struct cfi_private *retcfi;unsigned long *chip_map;int i, j, mapsize;int max_chips;memset(&cfi, 0, sizeof(cfi));/* Call the probetype-specific code with all permutations ofinterleave and device type, etc. */if (!genprobe_new_chip(map, cp, &cfi)) {/* The probe didn't like it */pr_debug("%s: Found no %s device at location zero\n",cp->name, map->name);return NULL;}...}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

genprobe_ident_chips->genprobe_new_chip执行cp->probe_chip:

static int genprobe_new_chip(struct map_info *map, struct chip_probe *cp,struct cfi_private *cfi)
{int min_chips = (map_bankwidth(map)/4?:1); /* At most 4-bytes wide. */int max_chips = map_bankwidth(map); /* And minimum 1 */int nr_chips, type;for (nr_chips = max_chips; nr_chips >= min_chips; nr_chips >>= 1) {...for (; type <= CFI_DEVICETYPE_X32; type<<=1) {...if (cp->probe_chip(map, 0, NULL, cfi))return 1;}}return 0;
}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

由于cp = &cfi_chip_probe 所以执行执行cp->probe_chip相当于执行cfi_chip_probe.probe_chip = cfi_probe_chip。
2、JEDEC mode:
①、注册jedec_chipdrv :

static int __init jedec_probe_init(void)
{register_mtd_chip_driver(&jedec_chipdrv);return 0;
}

在jedec_probe_init函数中调用register_mtd_chip_driver函数,将jedec_chipdrv 添加到chip_drvs_list中去:

void register_mtd_chip_driver(struct mtd_chip_driver *drv)
{spin_lock(&chip_drvs_lock);list_add(&drv->list, &chip_drvs_list);spin_unlock(&chip_drvs_lock);
}

其中,jedec_chipdrv 定义如下:

static struct mtd_chip_driver jedec_chipdrv = {.probe  = jedec_probe,.name   = "jedec_probe",.module = THIS_MODULE
};

②、do_map_probe执行过程:
在do_map_probe函数中调用get_mtd_chip_driver函数 :

struct mtd_info *do_map_probe(const char *name, struct map_info *map)
{struct mtd_chip_driver *drv;struct mtd_info *ret;drv = get_mtd_chip_driver(name);...}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

依次列举chip_drvs_list中的mtd_chip_driver 项,将每一项的成员name和传入参数name做比较,返回name相同的mtd_chip_driver 项,返回值ret = &jedec_chipdrv 给do_map_probe函数中的drv:

static struct mtd_chip_driver *get_mtd_chip_driver (const char *name)
{struct list_head *pos;struct mtd_chip_driver *ret = NULL, *this;spin_lock(&chip_drvs_lock);list_for_each(pos, &chip_drvs_list) {this = list_entry(pos, typeof(*this), list);if (!strcmp(this->name, name)) {ret = this;break;}}if (ret && !try_module_get(ret->module))ret = NULL;spin_unlock(&chip_drvs_lock);return ret; //ret = &jedec_chipdrv 
}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

再由mtd_chip_driver返回do_map_probe执行drv->probe(map):

struct mtd_info *do_map_probe(const char *name, struct map_info *map)
{struct mtd_chip_driver *drv;struct mtd_info *ret;...ret = drv->probe(map); //drv = &jedec_chipdrv ...return ret;
}
/* linux-3.4.2\drivers\mtd\chips\chipreg.c */

由于drv = &jedec_chipdrv ,执行执行drv->probe即相当于执行jedec_chipdrv .probe = jedec_probe:

static struct mtd_info *jedec_probe(struct map_info *map)
{/** Just use the generic probe stuff to call our CFI-specific* chip_probe routine in all the possible permutations, etc.*/return mtd_do_chip_probe(map, &jedec_chip_probe);
}
/* linux-3.4.2\drivers\mtd\chips\jedec_probe.c */

其中,jedec_chip_probe 定义如下:

static struct chip_probe jedec_chip_probe = {.name = "JEDEC",.probe_chip = jedec_probe_chip
};
/* linux-3.4.2\drivers\mtd\chips\jedec_probe.c */

jedec_probe->mtd_do_chip_probe:

struct mtd_info *mtd_do_chip_probe(struct map_info *map, struct chip_probe *cp)
{struct mtd_info *mtd = NULL;struct cfi_private *cfi;/* First probe the map to see if we have CFI stuff there. */cfi = genprobe_ident_chips(map, cp); //cp = &cfi_chip_probe ...}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

mtd_do_chip_probe->genprobe_ident_chips:

static struct cfi_private *genprobe_ident_chips(struct map_info *map, struct chip_probe *cp)
{struct cfi_private cfi;struct cfi_private *retcfi;unsigned long *chip_map;int i, j, mapsize;int max_chips;memset(&cfi, 0, sizeof(cfi));/* Call the probetype-specific code with all permutations ofinterleave and device type, etc. */if (!genprobe_new_chip(map, cp, &cfi)) {/* The probe didn't like it */pr_debug("%s: Found no %s device at location zero\n",cp->name, map->name);return NULL;}...}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

genprobe_ident_chips->genprobe_new_chip执行cp->probe_chip:

static int genprobe_new_chip(struct map_info *map, struct chip_probe *cp,struct cfi_private *cfi)
{int min_chips = (map_bankwidth(map)/4?:1); /* At most 4-bytes wide. */int max_chips = map_bankwidth(map); /* And minimum 1 */int nr_chips, type;for (nr_chips = max_chips; nr_chips >= min_chips; nr_chips >>= 1) {...for (; type <= CFI_DEVICETYPE_X32; type<<=1) {...if (cp->probe_chip(map, 0, NULL, cfi))return 1;}}return 0;
}
/* linux-3.4.2\drivers\mtd\chips\gen_probe.c */

由于cp = &jedec_chip_probe,所以执行执行cp->probe_chip相当于执行jedec_chip_probe.probe_chip =jedec_probe_chip 。

这篇关于NORFlash协议层的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

如何在Spring Boot项目中集成MQTT协议

《如何在SpringBoot项目中集成MQTT协议》本文介绍在SpringBoot中集成MQTT的步骤,包括安装Broker、添加EclipsePaho依赖、配置连接参数、实现消息发布订阅、测试接口... 目录1. 准备工作2. 引入依赖3. 配置MQTT连接4. 创建MQTT配置类5. 实现消息发布与订阅

使用Python进行GRPC和Dubbo协议的高级测试

《使用Python进行GRPC和Dubbo协议的高级测试》GRPC(GoogleRemoteProcedureCall)是一种高性能、开源的远程过程调用(RPC)框架,Dubbo是一种高性能的分布式服... 目录01 GRPC测试安装gRPC编写.proto文件实现服务02 Dubbo测试1. 安装Dubb

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优

关于WebSocket协议状态码解析

《关于WebSocket协议状态码解析》:本文主要介绍关于WebSocket协议状态码的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录WebSocket协议状态码解析1. 引言2. WebSocket协议状态码概述3. WebSocket协议状态码详解3

Qt 中集成mqtt协议的使用方法

《Qt中集成mqtt协议的使用方法》文章介绍了如何在工程中引入qmqtt库,并通过声明一个单例类来暴露订阅到的主题数据,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一,引入qmqtt 库二,使用一,引入qmqtt 库我是将整个头文件/源文件都添加到了工程中进行编译,这样 跨平台

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

【Go】go连接clickhouse使用TCP协议

离开你是傻是对是错 是看破是软弱 这结果是爱是恨或者是什么 如果是种解脱 怎么会还有眷恋在我心窝 那么爱你为什么                      🎵 黄品源/莫文蔚《那么爱你为什么》 package mainimport ("context""fmt""log""time""github.com/ClickHouse/clickhouse-go/v2")func main(

2024.9.8 TCP/IP协议学习笔记

1.所谓的层就是数据交换的深度,电脑点对点就是单层,物理层,加上集线器还是物理层,加上交换机就变成链路层了,有地址表,路由器就到了第三层网络层,每个端口都有一个mac地址 2.A 给 C 发数据包,怎么知道是否要通过路由器转发呢?答案:子网 3.将源 IP 与目的 IP 分别同这个子网掩码进行与运算****,相等则是在一个子网,不相等就是在不同子网 4.A 如何知道,哪个设备是路由器?答案:在 A

Modbus-RTU协议

一、协议概述 Modbus-RTU(Remote Terminal Unit)是一种基于主从架构的通信协议,采用二进制数据表示,消息中的每个8位字节含有两个4位十六进制字符。它主要通过RS-485、RS-232、RS-422等物理接口实现数据的传输,传输距离远、抗干扰能力强、通信效率高。 二、报文结构 一个标准的Modbus-RTU报文通常包含以下部分: 地址域:单个字节,表示从站设备