zoukankan      html  css  js  c++  java
  • volatile

    Why the "volatile" type class should not be used
    ------------------------------------------------
    
    C programmers have often taken volatile to mean that the variable could be
    changed outside of the current thread of execution; as a result, they are
    sometimes tempted to use it in kernel code when shared data structures are
    being used.  In other words, they have been known to treat volatile types
    as a sort of easy atomic variable, which they are not.  The use of volatile in
    kernel code is almost never correct; this document describes why.
    
    The key point to understand with regard to volatile is that its purpose is
    to suppress optimization, which is almost never what one really wants to
    do.  In the kernel, one must protect shared data structures against
    unwanted concurrent access, which is very much a different task.  The
    process of protecting against unwanted concurrency will also avoid almost
    all optimization-related problems in a more efficient way.
    
    Like volatile, the kernel primitives which make concurrent access to data
    safe (spinlocks, mutexes, memory barriers, etc.) are designed to prevent
    unwanted optimization.  If they are being used properly, there will be no
    need to use volatile as well.  If volatile is still necessary, there is
    almost certainly a bug in the code somewhere.  In properly-written kernel
    code, volatile can only serve to slow things down.
    
    Consider a typical block of kernel code:
    
        spin_lock(&the_lock);
        do_something_on(&shared_data);
        do_something_else_with(&shared_data);
        spin_unlock(&the_lock);
    
    If all the code follows the locking rules, the value of shared_data cannot
    change unexpectedly while the_lock is held.  Any other code which might
    want to play with that data will be waiting on the lock.  The spinlock
    primitives act as memory barriers - they are explicitly written to do so -
    meaning that data accesses will not be optimized across them.  So the
    compiler might think it knows what will be in shared_data, but the
    spin_lock() call, since it acts as a memory barrier, will force it to
    forget anything it knows.  There will be no optimization problems with
    accesses to that data.
    
    If shared_data were declared volatile, the locking would still be
    necessary.  But the compiler would also be prevented from optimizing access
    to shared_data _within_ the critical section, when we know that nobody else
    can be working with it.  While the lock is held, shared_data is not
    volatile.  When dealing with shared data, proper locking makes volatile
    unnecessary - and potentially harmful.
    
    The volatile storage class was originally meant for memory-mapped I/O
    registers.  Within the kernel, register accesses, too, should be protected
    by locks, but one also does not want the compiler "optimizing" register
    accesses within a critical section.  But, within the kernel, I/O memory
    accesses are always done through accessor functions; accessing I/O memory
    directly through pointers is frowned upon and does not work on all
    architectures.  Those accessors are written to prevent unwanted
    optimization, so, once again, volatile is unnecessary.
    
    Another situation where one might be tempted to use volatile is
    when the processor is busy-waiting on the value of a variable.  The right
    way to perform a busy wait is:
    
        while (my_variable != what_i_want)
            cpu_relax();
    
    The cpu_relax() call can lower CPU power consumption or yield to a
    hyperthreaded twin processor; it also happens to serve as a compiler
    barrier, so, once again, volatile is unnecessary.  Of course, busy-
    waiting is generally an anti-social act to begin with.
    
    There are still a few rare situations where volatile makes sense in the
    kernel:
    
      - The above-mentioned accessor functions might use volatile on
        architectures where direct I/O memory access does work.  Essentially,
        each accessor call becomes a little critical section on its own and
        ensures that the access happens as expected by the programmer.
    
      - Inline assembly code which changes memory, but which has no other
        visible side effects, risks being deleted by GCC.  Adding the volatile
        keyword to asm statements will prevent this removal.
    
      - The jiffies variable is special in that it can have a different value
        every time it is referenced, but it can be read without any special
        locking.  So jiffies can be volatile, but the addition of other
        variables of this type is strongly frowned upon.  Jiffies is considered
        to be a "stupid legacy" issue (Linus's words) in this regard; fixing it
        would be more trouble than it is worth.
    
      - Pointers to data structures in coherent memory which might be modified
        by I/O devices can, sometimes, legitimately be volatile.  A ring buffer
        used by a network adapter, where that adapter changes pointers to
        indicate which descriptors have been processed, is an example of this
        type of situation.
    
    For most code, none of the above justifications for volatile apply.  As a
    result, the use of volatile is likely to be seen as a bug and will bring
    additional scrutiny to the code.  Developers who are tempted to use
    volatile should take a step back and think about what they are truly trying
    to accomplish.
    
    Patches to remove volatile variables are generally welcome - as long as
    they come with a justification which shows that the concurrency issues have
    been properly thought through.
    
    
    NOTES
    -----
    
    [1] http://lwn.net/Articles/233481/
    [2] http://lwn.net/Articles/233482/
    
    CREDITS
    -------
    
    Original impetus and research by Randy Dunlap
    Written by Jonathan Corbet
    Improvements via comments from Satyam Sharma, Johannes Stezenbach, Jesper
    	Juhl, Heikki Orsila, H. Peter Anvin, Philipp Hahn, and Stefan
    	Richter.


    多线程编程中什么情况下需要加 volatile?

    网上说 volatile 有一种使用是作为「多线程被几个任务共享变量的修饰符」。但是我看过很多代码多线程临界区里面变量声明时并没有加 volatile。比方说印象中整个 muduo 网络库好像就 atomicInt 有 volatile;nginx 中一些时间的表示都用了 volatile。

    是否意味着在可重入(线程安全且不互斥使用)函数的访问中的变量才需要用 volatile 呢?但是似乎有些代码就是拿了个全局的 int 之类的当做多线程的标识变量,不加 volatile 可行么?

    总之
    • 多线程编程中什么情况下需要加 volatile?
    • 如果说 volatile 关键字对多线程无用,那为什么类似 muduo 和 nginx 的库都有用到呢?

    相关问题:
    C++多线程有必要加volatile么? - C++11
     
    按投票排序按时间排序

    21 个回答

    更多
    我来回答这个问题
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     

    写回答…

    加入知乎

    与世界分享你的知识、经验和见解

    验证码
    已有帐号?
     516 人关注该问题
    内页右上-圆桌-成为天文学家
     
     
     
  • 相关阅读:
    Redis Linux版安装详解
    cdh-hbase用户无法执行命令
    异常-java.util.concurrent.TimeoutException: Futures timed out after [100000 milliseconds]
    异常-Caused by: org.apache.hadoop.ipc.RemoteException(org.apache.hadoop.security.AccessControlException): Permission denied: user=hdfs, access=WRITE, inode="/hbase":root:supergroup:drwxr-xr-x
    【异常】Maxwell异常 Exception in thread "main" net.sf.jsqlparser.parser.TokenMgrError: Lexical error at line 1, column 596. Encountered: <EOF> after : ""
    【异常】jps6432 -- process information unavailable
    异常-Exception in thread "main" net.sf.jsqlparser.parser.TokenMgrError: Lexical error at line 1, column 596. Encountered: <EOF> after :
    【异常】java.sql.SQLException: Could not retrieve transaction read-only status from server Query
    【异常】ser class threw exception: java.sql.SQLException: The last packet successfully received from the server was 39,444 milliseconds ago. The last
    storedownloadd占用cpu高
  • 原文地址:https://www.cnblogs.com/liyulong1982/p/5522335.html
Copyright © 2011-2022 走看看