当前位置:  编程技术>软件工程/软件设计
本页文章导读:
    ▪Linux系统建立Nor Flash分区      Linux建立Nor Flash分区 韩大卫@吉林师范大学 接上文章<<linux系统Nor Flash芯片初始化及驱动>>, 当Nor Flash 芯片在flash芯片驱动器里链表chip_drvs_list中找到并调用名为”cfi_probe”的驱动后, .........
    ▪Linux下的多进程间共享资源的互斥访问      #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <fcntl.h> #include <sys/mman.h> #include <pthread.h> pthread_mutex_t* g_mutex; //创建共享的mutex void init_mu.........
    ▪一个简单的线程池      //threadpool.h #ifndef __THREADPOOL_H__ #define __THREADPOOL_H__ #include <pthread.h> typedef void* (*task_fun)(void*); //用链表来维护等待任务 typedef struct threadtask { //任务的执行函数 task.........

[1]Linux系统建立Nor Flash分区
    来源: 互联网  发布时间: 2013-11-19
Linux建立Nor Flash分区


韩大卫@吉林师范大学

接上文章<<linux系统Nor Flash芯片初始化及驱动>>, 当Nor Flash 芯片在flash芯片驱动器里链表chip_drvs_list中找到并调用名为”cfi_probe”的驱动后, 完成芯片初始化阶段, 接着进入linux对Flash建立分区阶段.


在 arch/mips/cavium-octeon/flash_setup.c 中

static struct map_info flash_map;


static int __init flash_init(void)

{                         
                 
    union cvmx_mio_boot_reg_cfgx region_cfg;
	
//从bootbus总线上获取flash的基地址.
    region_cfg.u64 = cvmx_read_csr(CVMX_MIO_BOOT_REG_CFGX(0));

    if (region_cfg.s.en) {
            
    //将全局数据结构struct map_info flash_map命名为octeon_nor0
        flash_map.name = "octeon_nor0";

	//物理地址和大小
        flash_map.phys = region_cfg.s.base << 16;

        flash_map.size = 0x1fc00000 - flash_map.phys;

        flash_map.bankwidth = 1;


	//使用ioremap()将32M 大小的Flash的物理地址映射到虚拟地址上.
        flash_map.virt = ioremap(flash_map.phys, flash_map.size);

        pr_notice("Bootbus flash: Setting flash for %luMB flash at "

              "0x%08llx\n", flash_map.size >> 20, flash_map.phys);

        simple_map_init(&flash_map);

	
		/*
	调用do_map_probe()进入Nor Flash芯片初始化阶段,该函数会在Flash芯片驱动器列表中找到名为cfi_probe的驱动器, 并调用其probe()函数, 准备好read/wirte/ioctl等函数的实现方法.
		*/
        mymtd = do_map_probe("cfi_probe", &flash_map);

        if (mymtd) {      

            mymtd->owner = THIS_MODULE;

     

#ifdef CONFIG_MTD_PARTITIONS

/*
	对FLash芯片成功探测(调用过probe)后, linux进入处理Flash分区阶段
*/
nr_parts = parse_mtd_partitions(mymtd,
part_probe_types,
&parts, 0);

            if (nr_parts > 0)


/*
	nr_parts>0 ,说明解析到存在多个分区, 那么添加各个分区
*/
                add_mtd_partitions(mymtd, parts, nr_parts);

            else          

                add_mtd_device(mymtd);

#else             

		//由于定义了CONFIG_MTD_PARTITIONS
宏, 不执行该函数        

            add_mtd_device(mymtd);

#endif                    

        } else {          

            pr_err("Failed to register MTD device for flash\n");

        }                 

    }                     

    return 0;             

}                                                

late_initcall(flash_init);                                                                                                                   

调用do_map_probe()后, 成功的话返回一个重要的数据结构struct mtd_info.

Struct mtd_info是linux描述MTD类型设备的数据结构, 里面有mtd设备等待初始化的信息(变量)和一些设备操作方法(函数指针).

经过do_map_probe()的初始化, 其中有一些成员已经得到赋值, 请参考drivers/mtd/chips/cfi_probe.c 中的cfi_probe()函数.



parse_mtd_partitions(mymtd,
 part_probe_types,&parts, 0); 


解析器类型: 有cmdlinepart和 RedBoot两中, 如没有定义CONFIG_MTD_REDBOOT_PARTS, 那只将cmdlinepart编译在内, 最后只连接cmdlinepart.o.

static const char *part_probe_types[] = { 

    "cmdlinepart",

#ifdef CONFIG_MTD_REDBOOT_PARTS

    "RedBoot",

#endif

    NULL

};


parse_mtd_partitions()定义在drivers/mtd/mtdpart.c 中:

int parse_mtd_partitions(struct mtd_info *master, const char **types,

             struct mtd_partition **pparts, unsigned long origin)

{

    struct mtd_part_parser *parser;

    int ret = 0;

 
	//循环次数: 解析器的个数, 这里只有一个解析器cmdlinepart, 故只循环一次

    for ( ; ret <= 0 && *types; types++) {


		//获取cmdlinepart解析器
        parser = get_partition_parser(*types);

        if (!parser && !request_module("%s", *types))

                parser = get_partition_parser(*types);

        if (!parser) {

            printk(KERN_NOTICE "%s partition parsing not available\n",

                   *types);

            continue;

        }   

		
		/*
		获取成功的话, 调用其解析函数 parse_fn(), 类似于do_map_probe()
		后者是获取驱动器cfi_probe, 获取成功的话, 调用其探测函数Probe()
		*/
        ret = (*parser->parse_fn)(master, pparts, origin);


		/*
			parse_fn()返回解析到的分区个数.在此打印出相关信息,此信息可在dmesg中看到.
		 parser->name 为cmdlinepart,  master->name为octeon_nor0
		*/
        if (ret > 0) {

            printk(KERN_NOTICE "%d %s partitions found on MTD device %s\n",

                   ret, parser->name, master->name);

         
    
[2]Linux下的多进程间共享资源的互斥访问
    来源: 互联网  发布时间: 2013-11-19
#include    <stdio.h>  
#include    <stdlib.h>  
#include    <unistd.h>  
#include    <fcntl.h>  
#include    <sys/mman.h>  
#include    <pthread.h>  
pthread_mutex_t* g_mutex;  
//创建共享的mutex  
void init_mutex(void)  
{  
    int ret;  
    //g_mutex一定要是进程间可以共享的,否则无法达到进程间互斥  
    g_mutex=(pthread_mutex_t*)mmap(NULL, sizeof(pthread_mutex_t), PROT_READ|PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, -1, 0);  
    if( MAP_FAILED==g_mutex )  
    {  
        perror("mmap");  
        exit(1);  
    }  
      
    //设置attr的属性  
    pthread_mutexattr_t attr;  
    pthread_mutexattr_init(&attr);  
    //一定要设置为PTHREAD_PROCESS_SHARED  
    //具体可以参考http://blog.chinaunix.net/u/22935/showart_340408.html  
    ret=pthread_mutexattr_setpshared(&attr,PTHREAD_PROCESS_SHARED);  
    if( ret!=0 )  
    {  
        perror("init_mutex pthread_mutexattr_setpshared");  
        exit(1);  
    }  
    pthread_mutex_init(g_mutex, &attr);  
}  
int main(int argc, char *argv[])  
{  
    init_mutex();  
    int ret;      
    char str1[]="this is child process/r/n";  
    char str2[]="this is father process/r/n";  
    int fd=open("tmp", O_RDWR|O_CREAT|O_TRUNC, 0666);  
    if( -1==fd )  
    {  
        perror("open");  
        exit(1);  
    }  
    pid_t pid;  
    pid=fork();  
    if( pid<0 )  
    {  
        perror("fork");  
        exit(1);  
    }  
    else if( 0==pid )  
    {  
        ret=pthread_mutex_lock(g_mutex);  
        if( ret!=0 )  
        {  
            perror("child pthread_mutex_lock");  
        }  
        sleep(10);//测试是否能够阻止父进程的写入  
        write(fd, str1, sizeof(str1));  
        ret=pthread_mutex_unlock(g_mutex);    
        if( ret!=0 )  
        {  
            perror("child pthread_mutex_unlock");  
        }     
    }  
    else  
    {  
        sleep(2);//保证子进程先执行   
        ret=pthread_mutex_lock(g_mutex);  
        if( ret!=0 )  
        {  
            perror("father pthread_mutex_lock");  
        }  
        write(fd, str2, sizeof(str2));  
        ret=pthread_mutex_unlock(g_mutex);    
        if( ret!=0 )  
        {  
            perror("father pthread_mutex_unlock");  
        }                 
    }  
    wait(NULL);  
    munmap(g_mutex, sizeof(pthread_mutex_t));  
}  


运行后tmp文件内容为:

this is child process

this is father process



作者:DLUTBruceZhang 发表于2013-4-22 14:30:30 原文链接
阅读:60 评论:0 查看评论

    
[3]一个简单的线程池
    来源: 互联网  发布时间: 2013-11-19
//threadpool.h  
#ifndef     __THREADPOOL_H__  
#define     __THREADPOOL_H__  
#include    <pthread.h>  
typedef void* (*task_fun)(void*);  
//用链表来维护等待任务  
typedef struct threadtask  
{  
    //任务的执行函数  
    task_fun task;  
    //执行函数的参数  
    void* arg;  
    //下一节点  
    struct threadtask* next;  
}THREAD_TASK;  
  
void* thr_fun(void* arg);//每个线程执行的函数  
int init_task(void);//初始化等待任务链表  
int init_pool(void);//初始化线程池  
//向任务链表中添加任务  
int add_task(task_fun task, void* arg);  
int destroy_poll(void);  
#endif  


//threadpool.c  
//对于基本不出现的错误不进行检测  
#include    <stdio.h>  
#include    <stdlib.h>  
#include    <unistd.h>  
#include    <string.h>  
#include    "threadpool.h"  
//线程池中最多允许的线程数  
#define     THREADPOOLSIZE 3  
//当前等待任务的个数  
static unsigned int taskwaittingnum=0;  
//taskwaittingnum的锁  
pthread_mutex_t g_nummutex= PTHREAD_MUTEX_INITIALIZER;  
//等待任务队列的锁  
pthread_mutex_t g_taskmutex=PTHREAD_MUTEX_INITIALIZER;  
THREAD_TASK* g_pTask=NULL;  
pthread_t tid[THREADPOOLSIZE];  
//是否销毁线程池  
static int isShut=0;  
void* thr_fun(void* arg)  
{  
    task_fun task;  
    void* funarg=NULL;  
    THREAD_TASK* ptmp=NULL;  
      
    while(1)  
    {  
        //如果要销毁线程池,跳出while循环  
        if( 1==isShut )   
        {  
            break;//跳出while循环  
        }  
        pthread_mutex_lock(&g_nummutex);  
        //如果当前没有任务需要调度,休眠5000微妙  
        if( 0==taskwaittingnum )  
        {  
            pthread_mutex_unlock(&g_nummutex);  
            usleep(5000);  
            continue;  
        }  
        //当前有任务调度  
        else  
        {  
            //需要写入链表,所以要锁定链表  
            pthread_mutex_lock(&g_taskmutex);  
            task=g_pTask->next->task;  
            funarg=g_pTask->next->arg;  
            ptmp=g_pTask->next;  
            g_pTask->next=ptmp->next;  
            taskwaittingnum--;  
            pthread_mutex_unlock(&g_taskmutex);  
            pthread_mutex_unlock(&g_nummutex);  
              
            free(ptmp);  
            ptmp=NULL;  
            (*task)(funarg);  
        }  
    }  
    pthread_exit(NULL);  
}  
//初始化任务链表,首个节点不用来存储任务  
int init_task(void)  
{  
    g_pTask=(THREAD_TASK*)malloc(sizeof(THREAD_TASK));  
    if( NULL==g_pTask )  
    {  
        printf("init_task malloc fails./n");  
        return -1;  
    }  
    memset(g_pTask, 0, sizeof(THREAD_TASK));  
    g_pTask->next=NULL;  
    return 0;  
}  
//初始化线程池  
int init_pool(void)  
{  
    int ret;  
    int i;  
    for( i=0 ; i<THREADPOOLSIZE ; i++ )  
    {  
        ret=pthread_create(tid+i, NULL, thr_fun, NULL);  
        if( ret!=0 )  
        {  
            printf("init_pool pthread_create:/n%s/n", strerror(ret));  
            return -1;  
        }  
    }  
    return 0;     
}  
int add_task(task_fun task, void* arg)  
{  
    THREAD_TASK* ptmp=NULL;  
    pthread_mutex_lock(&g_nummutex);  
    pthread_mutex_lock(&g_taskmutex);  
    ptmp=g_pTask;  
    while( ptmp->next!=NULL )  
    {  
        ptmp=ptmp->next;  
    }  
    ptmp->next=(THREAD_TASK*)malloc(sizeof(THREAD_TASK));  
    if( NULL==ptmp->next )  
    {  
        printf("add_task malloc fails/n");  
        return -1;  
    }  
    ptmp=ptmp->next;  
    ptmp->task=task;  
    ptmp->arg=arg;  
    ptmp->next=NULL;  
    taskwaittingnum++;  
    pthread_mutex_unlock(&g_nummutex);  
    pthread_mutex_unlock(&g_taskmutex);  
    return 0;  
}  
int destroy_pool(void)  
{  
    isShut=1;  
    pthread_mutex_destroy(&g_nummutex);  
    pthread_mutex_destroy(&g_taskmutex);  
    return 0;  
}  


//用来测试的main.c  
#include    <stdio.h>  
#include    "threadpool.h"  
void* task1(void* arg)  
{  
    printf("task1正在运行!/n");  
    sleep(10);  
    printf("task1运行结束!/n");  
    return NULL;  
}  
void* task2(void* arg)  
{  
    printf("task2正在运行!/n");  
    sleep(10);  
    printf("task2运行结束!/n");  
    return NULL;  
}  
void* task3(void* arg)  
{  
    printf("task3正在运行!/n");  
    printf("task3运行结束!/n");  
    return NULL;  
}  
void* task4(void* arg)  
{  
    printf("task4正在运行!/n");  
    printf("task4运行结束!/n");  
    return NULL;  
}  
int main(int argc, char *argv[])  
{  
    init_task();  
    init_pool();  
    add_task(task1, NULL);  
    add_task(task2, NULL);  
    add_task(task3, NULL);  
    add_task(task4, NULL);  
    sleep(30);  
    destroy_pool();  
    return 0;  
}  


作者:DLUTBruceZhang 发表于2013-4-22 14:28:01 原文链接
阅读:63 评论:0 查看评论

    
最新技术文章:
▪主-主数据库系统架构    ▪java.lang.UnsupportedClassVersionError: Bad version number i...    ▪eclipse项目出现红色叉叉解决方案
▪Play!framework 项目部署到Tomcat    ▪dedecms如何做中英文网站?    ▪Spring Batch Framework– introduction chapter(上)
▪第三章 AOP 基于@AspectJ的AOP    ▪基于插件的服务集成方式    ▪Online Coding开发模式 (通过在线配置实现一个表...
▪观察者模式(Observer)    ▪工厂模式 - 程序实现(java)    ▪几种web并行化编程实现
▪机器学习理论与实战(二)决策树    ▪Hibernate(四)——全面解析一对多关联映射    ▪我所理解的设计模式(C++实现)——解释器模...
▪利用规则引擎打造轻量级的面向服务编程模式...    ▪google blink的设计计划: Out-of-Progress iframes    ▪FS SIP呼叫的消息线程和状态机线程
▪XML FREESWITCH APPLICATION 实现    ▪Drupal 实战    ▪Blink: Chromium的新渲染引擎
▪(十四)桥接模式详解(都市异能版)    ▪你不知道的Eclipse用法:使用Allocation tracker跟...    ▪Linux内核-进程
▪你不知道的Eclipse用法:使用Metrics 测量复杂度    ▪IT行业为什么没有进度    ▪Exchange Server 2010/2013三种不同的故障转移
▪第二章 IoC Spring自动扫描和管理Bean    ▪CMMI简介    ▪目标检测(Object Detection)原理与实现(六)
▪值班总结(1)——探讨sql语句的执行机制    ▪第二章 IoC Annotation注入    ▪CentOS 6.4下安装Vagrant
▪Java NIO框架Netty1简单发送接受    ▪漫画研发之八:会吃的孩子有奶吃    ▪比较ASP和ASP.NET
▪SPRING中的CONTEXTLOADERLISTENER    ▪在Nginx下对网站进行密码保护    ▪Hibernate从入门到精通(五)一对一单向关联映...
▪.NET领域驱动设计—初尝(三:穿过迷雾走向光...    ▪linux下的块设备驱动(一)    ▪Modem项目工作总结
▪工作流--JBPM简介及开发环境搭建    ▪工作流--JBPM核心服务及表结构    ▪Eclipse:使用JDepend 进行依赖项检查
▪windows下用putty上传文件到远程Linux方法    ▪iBatis和Hibernate的5点区别    ▪基于学习的Indexing算法
▪设计模式11---设计模式之中介者模式(Mediator...    ▪带你走进EJB--JMS编程模型    ▪从抽象谈起(二):观察者模式与回调
▪设计模式09---设计模式之生成器模式(Builder)也...    ▪svn_resin_持续优化中    ▪Bitmap recycle方法与制作Bitmap的内存缓存
▪Hibernate从入门到精通(四)基本映射    ▪设计模式10---设计模式之原型模式(Prototype)    ▪Dreamer 3.0 支持json、xml、文件上传
▪Eclipse:使用PMD预先检测错误    ▪Jspx.net Framework 5.1 发布    ▪从抽象谈起(一):工厂模式与策略模式
▪Eclipse:使用CheckStyle实施编码标准    ▪【论文阅读】《Chain Replication for Supporting High T...    ▪Struts2 Path_路径问题
▪spring 配置文件详解    ▪Struts2第一个工程helloStruts极其基本配置    ▪Python学习入门基础教程(learning Python)--2 Python简...
▪maven springmvc环境配置    ▪基于SCRUM的金融软件开发项目    ▪software quality assurance 常见问题收录
▪Redis集群明细文档    ▪Dreamer 框架 比Struts2 更加灵活    ▪Maven POM入门
▪git 分支篇-----不断更新中    ▪Oracle非主键自增长    ▪php设计模式——UML类图
▪Matlab,Visio等生成的图片的字体嵌入问题解决...    ▪用Darwin和live555实现的直播框架    ▪学习ORM框架—hibernate(二):由hibernate接口谈...
▪(十)装饰器模式详解(与IO不解的情缘)    ▪无锁编程:最简单例子    ▪【虚拟化实战】网络设计之四Teaming
▪OSGi:生命周期层    ▪Javascript/Jquery——简单定时器    ▪java代码 发送GET、POST请求
▪Entity Framework底层操作封装(3)    ▪HttpClient 发送GET、POST请求    ▪使用spring框架,应用启动时,加载数据
▪Linux下Apache网站目录读写权限的设置    ▪单键模式的C++描述    ▪学习ORM框架—hibernate(一):初识hibernate
 


站内导航:


特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

©2012-2021,,E-mail:www_#163.com(请将#改为@)

浙ICP备11055608号-3