成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java開源工具在linux上的源碼分析(四):safe point

開發 后端
safe point 顧明思意,就是安全點,當需要jvm做一些操作的時候,需要把當前正在運行的線程進入一個安全點的狀態(也可以說停止狀態),這樣才能做一些安全的操作,比如線程的dump,堆棧的信息。

safe point 顧明思意,就是安全點,當需要jvm做一些操作的時候,需要把當前正在運行的線程進入一個安全點的狀態(也可以說停止狀態),這樣才能做一些安全的操作,比如線程的dump,堆棧的信息。

在jvm里面通常vm_thread(我們一直在談論的做一些屬于vm 份內事情的線程) 和cms_thread(內存回收的線程)做的操作,是需要將其他的線程通過調用SafepointSynchronize::begin 和 SafepointSynchronize:end來實現讓其他的線程進入或者退出safe point 的狀態。

通常safepoint 的有三種狀態

 

_not_synchronized

說明沒有任何打斷現在所有線程運行的操作,也就是vm thread, cms thread 沒有接到操作的指令

_synchronizing

vm thread,cms thread 接到操作指令,正在等待所有線程進入safe point

_synchronized

所有線程進入safe point, vm thread, cms thread 可以開始指令操作

 

Java線程的狀態

通常在java 進程中的Java 的線程有幾個不同的狀態,如何讓這些線程進入safepoint 的狀態中,jvm是采用不同的方式

a. 正在解釋執行

由于java是解釋性語言,而線程在解釋java 字節碼的時候,需要dispatch table,記錄方法地址進行跳轉的,那么這樣讓線程進入停止狀態就比較容易了,只要替換掉dispatch table 就可以了,讓線程知道當前進入softpoint 狀態。

java里會設置3個DispatchTable,  _active_table,  _normal_table, _safept_table

_active_table 正在解釋運行的線程使用的dispatch table

_normal_table 就是正常運行的初始化的dispatch table

_safept_table safe point需要的dispatch table

解釋運行的線程一直都在使用_active_table,關鍵處就是在進入saftpoint 的時候,用_safept_table替換_active_table, 在退出saftpoint 的時候,使用_normal_table來替換_active_table。

具體實現可以查看源碼

  1. void TemplateInterpreter::notice_safepoints() {  
  2.   if (!_notice_safepoints) {  
  3.     // switch to safepoint dispatch table  
  4.     _notice_safepoints = true;  
  5.     copy_table((address*)&_safept_table, (address*)&_active_table, sizeof(_active_table) / sizeof(address));  
  6.   }  
  7. }  
  8.  
  9. // switch from the dispatch table which notices safepoints back to the  
  10. // normal dispatch table.  So that we can notice single stepping points,  
  11. // keep the safepoint dispatch table if we are single stepping in JVMTI.  
  12. // Note that the should_post_single_step test is exactly as fast as the  
  13. // JvmtiExport::_enabled test and covers both cases.  
  14. void TemplateInterpreter::ignore_safepoints() {  
  15.   if (_notice_safepoints) {  
  16.     if (!JvmtiExport::should_post_single_step()) {  
  17.       // switch to normal dispatch table  
  18.       _notice_safepoints = false;  
  19.       copy_table((address*)&_normal_table, (address*)&_active_table, sizeof(_active_table) / sizeof(address));  
  20.     }  
  21.   }  

b. 運行在native code

如果線程運行在native code的時候,vm thread 是不需要等待線程執行完的,只需要在從native code 返回的時候去判斷一下 _state 的狀態就可以了。

在方法體里就是前面博客也出現過的 SafepointSynchronize::do_call_back()

  1. inline static bool do_call_back() {  
  2.   return (_state != _not_synchronized);  

判斷了_state 不是_not_synchronized狀態

為了能讓線程從native code 回到java 的時候為了能讀到/設置正確線程的狀態,通常的解決方法使用memory barrier,java 使用OrderAccess::fence(); 在匯編里使用__asm__ volatile ("lock; addl $0,0(%%rsp)" : : : "cc", "memory"); 保證從內存里讀到正確的值,但是這種方法嚴重影響系統的性能,于是java使用了每個線程都有獨立的內存頁來設置狀態。通過使用使用參數-XX:+UseMembar 參數使用memory barrier,默認是不打開的,也就是使用獨立的內存頁來設置狀態。

c. 運行編譯的代碼

1. Poling page 頁面

Poling page是在jvm初始化啟動的時候會初始化的一個單獨的內存頁面,這個頁面是讓運行的編譯過的代碼的線程進入停止狀態的關鍵。

在linux里面使用了mmap初始化,源碼如下

  1. address polling_page = (address) ::mmap(NULL, Linux::page_size(), PROT_READ, MAP_PRIVATE|MAP_ANONYMOUS, -10); 

2. 編譯

java 的JIT 會直接編譯一些熱門的源碼到機器碼,直接執行而不需要在解釋執行從而提高效率,在編譯的代碼中,當函數或者方法塊返回的時候會去訪問一個內存poling頁面。

x86架構下

  1. void LIR_Assembler::return_op(LIR_Opr result) {  
  2.   assert(result->is_illegal() || !result->is_single_cpu() || result->as_register() == rax, "word returns are in rax,");  
  3.   if (!result->is_illegal() && result->is_float_kind() && !result->is_xmm_register()) {  
  4.     assert(result->fpu() == 0"result must already be on TOS");  
  5.   }  
  6.  
  7.   // Pop the stack before the safepoint code  
  8.   __ remove_frame(initial_frame_size_in_bytes());  
  9.  
  10.   bool result_is_oop = result->is_valid() ? result->is_oop() : false;  
  11.  
  12.   // Note: we do not need to round double result; float result has the right precision  
  13.   // the poll sets the condition code, but no data registers  
  14.   AddressLiteral polling_page(os::get_polling_page() + (SafepointPollOffset % os::vm_page_size()),  
  15.                               relocInfo::poll_return_type);  
  16.  
  17.   // NOTE: the requires that the polling page be reachable else the reloc  
  18.   // goes to the movq that loads the address and not the faulting instruction  
  19.   // which breaks the signal handler code  
  20.  
  21.   __ test32(rax, polling_page);  
  22.  
  23.   __ ret(0);  

在前面提到的SafepointSynchronize::begin 函數源碼中

  1. if (UseCompilerSafepoints && DeferPollingPageLoopCount < 0) {  
  2.   // Make polling safepoint aware  
  3.   guarantee (PageArmed == 0"invariant") ;  
  4.   PageArmed = 1 ;  
  5.   os::make_polling_page_unreadable();  

這里提到了2個參數 UseCompilerSafepoints 和 DeferPollingPageLoopCount ,在默認的情況下這2個參數是true和-1

函數體將會調用os:make_polling_page_unreadable();在linux os 下具體實現是調用了mprotect(bottom,size,prot) 使polling 內存頁變成不可讀。

3. 信號

到當編譯好的程序嘗試在去訪問這個不可讀的polling頁面的時候,在系統級別會產生一個錯誤信號SIGSEGV, 可以參考筆者的一篇博客中曾經講過java 的信號處理,可以知道信號SIGSEGV的處理函數在x86體系下見下源碼:

  1. JVM_handle_linux_signal(int sig,  
  2.                         siginfo_t* info,  
  3.                         void* ucVoid,  
  4.                         int abort_if_unrecognized){  
  5.    ....  
  6.    if (sig == SIGSEGV && os::is_poll_address((address)info->si_addr)) {  
  7.         stub = SharedRuntime::get_poll_stub(pc);  
  8.       }   
  9.    ....  

在linux x86,64 bit的體系中,poll stub 的地址 就是 SafepointSynchronize::handle_polling_page_exception 詳細程序可見shareRuntime_x86_64.cpp

回到safepoint.cpp中,SafepointSynchronize::handle_polling_page_exception通過取出線程的safepoint_stat,調用函數void ThreadSafepointState::handle_polling_page_exception,***通過調用SafepointSynchronize::block(thread()); 來block當前線程。

 

d. block 狀態

當線程進入block狀態的時候,繼續保持block狀態。

原文鏈接:http://blog.csdn.net/raintungli/article/details/7162468

【系列文章】

  1. Java開源工具在linux上的源碼分析(一):跟蹤方式
  2. Java開源工具在linux上的源碼分析(二):信號處理
  3. Java開源工具在linux上的源碼分析(三):執行的線程vm thread
  4. Java開源工具在linux上的源碼分析(五):-F參數的bug
  5. Java開源工具在linux上的源碼分析(六):符號表的讀取
責任編輯:林師授 來源: raintungli的博客
相關推薦

2012-03-02 12:14:19

JavaJstackJmap

2012-03-02 12:20:21

Javajmapjstack

2012-03-02 12:25:07

Javajmapjstack

2012-03-02 12:38:49

Javajmapjstack

2012-03-02 13:29:38

Javajmapjstack

2022-06-26 18:09:43

Linux開源

2019-10-16 17:00:51

LinuxUbuntuVMware

2020-05-09 12:01:40

Linux開源軟件SDN

2022-02-18 15:19:52

日志收集工具開源

2010-01-27 09:58:59

Linuxunix程序日志

2021-03-09 11:25:04

Linux開源工具服務器

2018-05-30 09:00:00

2019-08-01 09:52:46

LinuxNetData性能監控工具

2012-05-22 00:28:21

JavaJava開源開源工具

2021-08-31 09:41:57

LinuxiPhone開源工具

2021-09-01 09:47:25

Linux 工具 開發

2022-06-06 14:20:25

個人財務開源預算

2019-05-23 14:36:24

LinuxSOSReportxsos

2021-11-25 09:25:51

Linux服務器開源工具

2017-01-12 15:58:17

Linux死鎖分析方法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91精品国产91久久久久久丝袜 | 黄免费观看| 久久久久亚洲精品 | 成人免费网站视频 | 国产偷自视频区视频 | 久久99精品视频 | 91美女在线 | 久久久久久国产精品 | 欧美日产国产成人免费图片 | 国产一级视频在线观看 | 国产精品久久久久久久久久免费看 | 精品伊人久久 | 久久精品视频在线观看 | 日韩中文字幕在线观看 | 亚洲天堂av在线 | 国产成人99久久亚洲综合精品 | 成人在线视频网 | 国产欧美一区二区三区久久 | 中文字幕国| 91av视频在线观看 | 天天干天天操天天射 | 欧美美女一区二区 | 久久综合国产精品 | 免费观看的av毛片的网站 | 97伦理最新伦理 | 羞羞视频免费观看 | 伊人性伊人情综合网 | 国产日韩欧美电影 | 精品91| 成人三级影院 | 久久久久国产精品一区二区 | 国产精品免费看 | 精品国产色 | 综合九九 | 色婷婷久久 | 伊人在线视频 | 国产精品免费福利 | 国产精品久久久久久久久久免费 | 国产在线www| 国产色在线 | 污污的网站在线观看 |