临界区问题以和解决方法.docVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
临界区问题的产生,以及解决方法。 临界区问题是嵌入式软件编程一个不得不面对的关键性问题。特别对于底层驱动,代码在内存中只有一份,上层的多任务或者多进程,都会对同一个驱动去访问,这样不可避免的遇到了任务之间打架的问题,处理好这个问题是区分一个菜鸟和老鸟的根本性关键之一。 接下来谈谈临界区产生的原因: 假设有以下代码: int x; void process_data() { x++; } 假如在一个可以抢占的操作系统上有两个任务task1, task2, 全局变量x 的初始值为0, 现在两个任务task1, task2 同时去访问process_data 这个函数,两个任务各执行一次process_data 这个函数,等到两个人执行完毕后,试问x的值是多少?大部分人可能会回答为2。没有操作系统的时候,的却不错,调用函数2次,就是2.问题是有了操作系统就没这么简单了,一个任务执行期间,随时可能会被另外一个任务给打断,这样就会造成临界区的问题。 首先明确一个基本概念,在操作系统中每一个任务都有自己的一套寄存器,各个任务间的寄存器值很可能是不一样的。 下面来具体分析这个问题产生的根本原因: x++不是一个原子型的操作,它的汇编函数有3句,分别是: 1 ldr r1, [mem] 2 add r1, r1, #1 3 strr1 [mem] 如果有以下流程,参照下图: 假如任务task1 刚执行完(2)即 add r1, r1,#1, 因为是可以抢占的操作系统,所以被高优先级任务task 2 给抢占了,然后task 2 执行完(3) (4) (5)这三个步骤之后还给任务task 1, 最后task1 执行完(6)。 如前所述,图中的task1 和task2 的寄存器值是不同的,因为任务各自有自己的一套寄存器。读者可以推导一下,x 的最终值在内存中是1而不是2! 所以在多任务的情况下,共同去访问一个全局变量,会产生临界区的问题,如之前所述最终值可能是不确定的,可能是1也可能是2,所以需要采用操作系统的一定机制去保护它。 接下来谈谈怎么去解决这个问题。 解决方式一: void process_data() { RAW_CPU_DISABLE(); x++; RAW_CPU_ENABLE(); } 如上代码关了中断的话,任务也就不能被抢了,而且x++的速度很快,推荐使用这样的方式。 解决方式二: void __process_data() { x++; } void process_data() { raw_disable_sche(); __process_data(); raw_enable_sche(); } 如上代码关了系统抢占后,任务之间的调度被禁止。 解决方式三: void __process_data() { x++; } void process_data() { lock(); __process_data(); unlock (); } 如上代码加软件锁之后,只能有一个任务处理此段临界区,lock可以是semaphore 或者mutex. 之前的例子演示的是任务和任务之间的打架冲突,会造成临界区问题,那如果是中断和任务之间的打架冲突呢?答案是一致的,唯一不同点是任务和任务之间的冲突可以有多种保护方式,但是任务和中断之间的冲突只能用关中断去保护,即采用之前的解决方式一来解决冲突。 临界区的产生原因有很多,往往是复杂的,考虑如下的fifo循环缓冲区: struct raw_fifo { RAW_U32 in; (1) RAW_U32 out; (2) RAW_U32 mask; void *data; RAW_U32 free_bytes; RAW_U32 size; }; 处是fifo的的写指针,(2)处是fifo读指针 下面的代码是往fifo 里面写数据。 RAW_U32 fifo_in(struct raw_fifo *fifo, const void *buf, RAW_U32 len) { RAW_U32 l; RAW_SR_ALLOC(); RAW_CPU_DISABLE(); (1) l = fifo_unused(fifo); if (len l) len = l; fifo_copy_in(fifo, buf, len, fifo-in); (2) fifo-in += len;

文档评论(0)

sy78219 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档