国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

鴻蒙輕內核A核源碼分析系列二:數據結構-位圖操作

開發 前端
文章由鴻蒙社區產出,想要了解更多內容請前往:51CTO和華為官方戰略合作共建的鴻蒙技術社區https://harmonyos.51cto.com

[[405939]]

想了解更多內容,請訪問:

51CTO和華為官方合作共建的鴻蒙技術社區

https://harmonyos.51cto.com

在進一步分析之前,本文我們先來熟悉下OpenHarmony鴻蒙輕內核提供的位操作模塊,在互斥鎖等模塊對位操作有使用。位操作是指對二進制數的bit位進行操作。程序可以設置某一變量為狀態字,狀態字中的每一bit位(標志位)可以具有自定義的含義。

1 位操作的宏定義

位操作模塊提供對32位無符號整數數值的bit位進行操作,bit位取值為0-31,以0開始計算,從左向右,第0位,第1位。。。第31位等。⑴處定義的宏OS_BITMAP_MASK如下,也就是十進制31。如果傳入的比特位pos大于31,會通過邏輯與運算截斷(pos & OS_BITMAP_MASK),只取低5位,確保不會大于31,避免溢出。⑵處定義的位圖掩碼全是1。

  1. ⑴  #define OS_BITMAP_MASK 0x1FU 
  2. ⑵  #define OS_BITMAP_WORD_MASK ~0UL 

 在文件kernel\include\los_bitmap.h中定義了常用的位操作相關的宏。宏BITMAP_WORD根據參數x計算出需要操作第幾個狀態字,由于計算狀態字的使用的是UINTPTR,狀態字可以是32位、也可以是64位。后文,我們默認以32位進行講解。宏BITMAP_FIRST_WORD_MASK傳入的參數是位操作的開始bit位數,用于計算需要進行位操作的掩碼,從開始位全部是1,宏BITMAP_LAST_WORD_MASK傳入的參數是位操作的結束bit位數,用于計算需要進行位操作的掩碼,結束位之前全部是1。宏BITMAP_NUM_WORDS傳入位數,計算狀態字的數量。

  1. #define _ONE(x) (1 + ((x) - (x))) 
  2.  #define BIT(n)  (1U << (n)) 
  3.  #define BIT_GET(x, bit) ((x) & (_ONE(x) << (bit))) 
  4.  #define BIT_SHIFT(x, bit) (((x) >> (bit)) & 1) 
  5.  #define BITS_GET(x, high, low) ((x) & (((_ONE(x) << ((high) + 1)) - 1) & ~((_ONE(x) << (low)) - 1))) 
  6.  #define BITS_SHIFT(x, high, low) (((x) >> (low)) & ((_ONE(x) << ((high) - (low) + 1)) - 1)) 
  7.  #define BIT_SET(x, bit) (((x) & (_ONE(x) << (bit))) ? 1 : 0) 
  8.  #define BITMAP_BITS_PER_WORD (sizeof(UINTPTR) * 8) 
  9.  #define BITMAP_NUM_WORDS(x) (((x) + BITMAP_BITS_PER_WORD - 1) / BITMAP_BITS_PER_WORD) 
  10.  #define BITMAP_WORD(x) ((x) / BITMAP_BITS_PER_WORD) 
  11.  #define BITMAP_BIT_IN_WORD(x) ((x) & (BITMAP_BITS_PER_WORD - 1)) 
  12.  #define BITMAP_FIRST_WORD_MASK(start) (~0UL << ((start) % BITMAP_BITS_PER_WORD)) 
  13.  #define BITMAP_LAST_WORD_MASK(nbits) \ 
  14.      (((nbits) % BITMAP_BITS_PER_WORD) ? (1UL << ((nbits) % BITMAP_BITS_PER_WORD)) - 1 : ~0UL) 
  15.  #define BITMAP_BITS_PER_INT (sizeof(INTPTR) * 8) 
  16.  #define BITMAP_BIT_IN_INT(x) ((x) & (BITMAP_BITS_PER_INT - 1)) 
  17.  #define BITMAP_INT(x) ((x) / BITMAP_BITS_PER_INT) 
  18.  #define BIT_MASK(x) (((x) >= sizeof(UINTPTR) * 8) ? (0UL - 1) : ((1UL << (x)) - 1)) 

2 位操作常用功能

OpenHarmony鴻蒙輕內核的位操作模塊提供標志位的置1和清0操作,可以改變標志位的內容,同時還提供獲取狀態字中標志位為1的最高位和最低位的功能。用戶也可以對系統的寄存器進行位操作。位操作提供了7個API,進行置1、清0、獲取為1的最高、最低位等操作,如下:

下面,我們剖析下位操作的源代碼。

2.1 LOS_BitmapSet()對狀態字的某一標志位進行置1操作

對狀態字的某一標志位進行置1操作。我們先看看傳入的參數,需要的2個參數分別是:需要改變bit位內容的狀態字UINT32 *bitmap,需要改變的bit位位數UINT16 pos。

代碼很簡單,首先進行基礎的校驗,如果狀態字為空,則返回。然后計算pos & OS_BITMAP_MASK,只取二進制的低5位,最大位值為31,避免左移的時候發生溢出。1U << (pos & OS_BITMAP_MASK)就是需要改變內容的狀態字的bit位,通過按位或運算設置狀態字UINT32 *bitmap的指定bit位的內容為1。

  1. VOID LOS_BitmapSet(UINT32 *bitmap, UINT16 pos) 
  2.     if (bitmap == NULL) { 
  3.         return
  4.     } 
  5.  
  6.     *bitmap |= 1U << (pos & OS_BITMAP_MASK); 

2.2 LOS_BitmapClr()對狀態字的某一標志位進行清0操作

對狀態字的某一標志位進行清0操作,代碼和置1操作對應,比較簡單,~(1U << (pos & OS_BITMAP_MASK))表示需要改變內容的狀態字的bit位為0,其余位為1,然后通過按位與運算設置狀態字UINT32 *bitmap的指定bit位的內容為0。

  1. VOID LOS_BitmapClr(UINT32 *bitmap, UINT16 pos) 
  2.     if (bitmap == NULL) { 
  3.         return
  4.     } 
  5.  
  6.     *bitmap &= ~(1U << (pos & OS_BITMAP_MASK)); 

2.3 LOS_HighBitGet()獲取狀態字中為1的最高位

代碼中CLZ(bitmap)是宏,展開為(__builtin_clz(bitmap)),這是編譯器內置的高效位運算的庫函數,clz是count leading zeros的縮寫,就是統計二進制數值中高位區開頭的全是0的數目。使用OS_BITMAP_MASK減去該值,結果就是狀態字中的1的最高位。

  1. UINT16 LOS_HighBitGet(UINT32 bitmap) 
  2.     if (bitmap == 0) { 
  3.         return LOS_INVALID_BIT_INDEX; 
  4.     } 
  5.  
  6.     return (OS_BITMAP_MASK - CLZ(bitmap)); 

2.4 LOS_LowBitGet()獲取狀態字中為1的最低位

代碼其中CTZ(bitmap)是宏,展開為(__builtin_ctz(value)),這是編譯器內置的高效位運算的庫函數,ctz是count trailing zeros的縮寫,就是統計二進制數值中低位區結尾的全是0的數目,該結果就是狀態字中的1的最低位。

  1. UINT16 LOS_LowBitGet(UINT32 bitmap) 
  2.     if (bitmap == 0) { 
  3.         return LOS_INVALID_BIT_INDEX; 
  4.     } 
  5.  
  6.     return CTZ(bitmap); 

2.5 LOS_BitmapSetNBits()對狀態字的連續標志位進行置1操作

可以使用LOS_BitmapSetNBits()函數對狀態字的連續比特位進行置1操作,第一個參數是需要改變bit位內容的狀態字UINT32 *bitmap,第二個參數是需要置1的bit位開始數start,第三個參數是需要置1的數量numsSet。由于bit位開始數start并沒有限制在[0,31],所以實際上設置的可能是UINT32 *bitmap狀態字后面的狀態字,需要根據業務實際情況進行設置,避免覆寫其他內存。同樣,需要置1的數量numsSet也可能跨多個狀態字。如圖所示:

我們看下代碼,

⑴處計算出需要操作的狀態字,其中BITMAP_WORD(start)計算相對狀態字bitmap需要偏移的數量,如果start處于區間[0,31],BITMAP_WORD(start)等于0,操作的就是狀態字bitmap。如果start處于區間[32,63],BITMAP_WORD(start)等于1,操作的就是狀態字bitmap后面的第一個狀態字,以此類推。

⑵處size可以和bit位開始數start結合來理解,size就是需要置1的bit位結束位數。

⑶處需要置1操作的bit位的位數。

⑷是對應需要置1操作的bit位的掩碼。

⑸處如果條件成立,說明需要置1操作需要跨多個狀態字進行操作,代碼會一個狀態字處理完畢,再去處理下一個狀態字。

⑹處把當前狀態字的相應的bit位進行置1操作,然后執行⑺把剩余需要置1的位數減去已經置1的位數。

⑻處更新bitsToSet和maskToSet,然后指針p指向下一個狀態字。

⑼處如果需要置1的位數大于0,并且此時已經可以在一個狀態字內完成操作,執行⑽處計算需要置1操作的掩碼,從bit開始位到結束位需要進行置1。

⑾處代碼執行置1操作,完成對狀態字的連續標志位進行置1操作。

  1. VOID LOS_BitmapSetNBits(UINTPTR *bitmap, UINT32 start, UINT32 numsSet) 
  2. ⑴  UINTPTR *p = bitmap + BITMAP_WORD(start); 
  3. ⑵  const UINT32 size = start + numsSet; 
  4. ⑶  UINT16 bitsToSet = BITMAP_BITS_PER_WORD - (start % BITMAP_BITS_PER_WORD); 
  5. ⑷  UINTPTR maskToSet = BITMAP_FIRST_WORD_MASK(start); 
  6.  
  7. ⑸  while (numsSet > bitsToSet) { 
  8. ⑹      *p |= maskToSet; 
  9. ⑺      numsSet -= bitsToSet; 
  10. ⑻      bitsToSet = BITMAP_BITS_PER_WORD; 
  11.         maskToSet = OS_BITMAP_WORD_MASK; 
  12.         p++; 
  13.     } 
  14. ⑼  if (numsSet) { 
  15. ⑽      maskToSet &= BITMAP_LAST_WORD_MASK(size); 
  16.         *p |= maskToSet; 
  17.     } 

2.6 LOS_BitmapClrNBits()對狀態字的連續標志位進行清0操作

可以使用LOS_BitmapClrNBits()函數對狀態字的連續比特位進行清0操作,第一個參數是需要改變bit位內容的狀態字UINT32 *bitmap,第二個參數是需要清0的bit位開始數start,第三個參數是需要清0的數量numsClear。該函數是函數LOS_BitmapSetNBits()的反向操作,代碼解釋可以參考函數LOS_BitmapSetNBits()。

  1. VOID LOS_BitmapClrNBits(UINTPTR *bitmap, UINT32 start, UINT32 numsClear) 
  2.     UINTPTR *p = bitmap + BITMAP_WORD(start); 
  3.     const UINT32 size = start + numsClear; 
  4.     UINT16 bitsToClear = BITMAP_BITS_PER_WORD - (start % BITMAP_BITS_PER_WORD); 
  5.     UINTPTR maskToClear = BITMAP_FIRST_WORD_MASK(start); 
  6.  
  7.     while (numsClear >= bitsToClear) { 
  8.         *p &= ~maskToClear; 
  9.         numsClear -= bitsToClear; 
  10.         bitsToClear = BITMAP_BITS_PER_WORD; 
  11.         maskToClear = OS_BITMAP_WORD_MASK; 
  12.         p++; 
  13.     } 
  14.     if (numsClear) { 
  15.         maskToClear &= BITMAP_LAST_WORD_MASK(size); 
  16.         *p &= ~maskToClear; 
  17.     } 

2.7 LOS_BitmapFfz()獲取從最低有效位開始的第一個0的bit位

可以使用LOS_BitmapFfz()函數獲取從最低有效位開始的第一個0的bit位位數,第一個參數是需要改變bit位內容的狀態字UINT32 *bitmap,第二個參數numBits表示最大的位數,對返回值進行限制,需要在指定的位數內找到符合條件的位數,否則返回-1。

在看函數代碼之前,先了解下Ffz()函數,如下:調用內嵌函數__builtin_ffsl()可以獲取一個unsigned long類型數字的二進制形式的從左開始的第一個1的位數,這個位數從1開始計數。比如對于二進制數字0110,該函數會返回2。在下面的函數中,給函數__builtin_ffsl()傳入的參數進行了取反,并減去了1,所以Ffz()函數返回一個數字從左開始的第一個0的位數,這個位數從0開始計數。

  1. /* find first zero bit starting from LSB */ 
  2. STATIC INLINE UINT16 Ffz(UINTPTR x) 
  3.     return __builtin_ffsl(~x) - 1; 

我們接著看下函數LOS_BitmapFfz()的代碼。⑴處根據位數numBits計算出對應的狀態字的數量,然后依次循環每一個狀態字,⑵處如果狀態字全為1,則繼續循環,否則執行⑶。執行到⑶說明,,前面有i個狀態字的各個位全為1。i * BITMAP_BITS_PER_WORD + Ffz(bitmap[i])就表示各個狀態字的二進制位中,從左到右第一個0的位置。⑷處如果獲取的位數小于第二個參數,則返回獲取的位數,否則返回-1。如下圖所示:

源代碼如下:

  1. INT32 LOS_BitmapFfz(UINTPTR *bitmap, UINT32 numBits) 
  2.     INT32 bit, i; 
  3.  
  4. ⑴  for (i = 0; i < BITMAP_NUM_WORDS(numBits); i++) { 
  5. ⑵      if (bitmap[i] == OS_BITMAP_WORD_MASK) { 
  6.             continue
  7.         } 
  8. ⑶      bit = i * BITMAP_BITS_PER_WORD + Ffz(bitmap[i]); 
  9. ⑷      if (bit < numBits) { 
  10.             return bit
  11.         } 
  12.         return -1; 
  13.     } 
  14.     return -1; 

小結

本文帶領大家一起剖析了鴻蒙輕內核的位操作模塊的源代碼。

想了解更多內容,請訪問:

51CTO和華為官方合作共建的鴻蒙技術社區

https://harmonyos.51cto.com

 

責任編輯:jianghua 來源: 鴻蒙社區
相關推薦

2021-05-12 09:45:20

鴻蒙HarmonyOS應用

2021-05-10 15:05:56

鴻蒙HarmonyOS應用

2021-04-30 15:06:34

鴻蒙HarmonyOS應用

2021-11-08 15:06:15

鴻蒙HarmonyOS應用

2022-03-11 20:23:14

鴻蒙源碼分析進程管理

2022-04-13 11:02:12

鴻蒙事件模塊事件Event

2022-03-03 18:28:28

Harmony進程任務管理模塊

2022-01-12 10:50:23

鴻蒙HarmonyOS應用

2022-01-10 15:31:44

鴻蒙HarmonyOS應用

2021-06-04 09:57:49

鴻蒙HarmonyOS應用

2021-05-17 09:28:59

鴻蒙HarmonyOS應用

2021-05-08 15:14:50

鴻蒙HarmonyOS應用

2021-06-04 14:15:10

鴻蒙HarmonyOS應用

2021-11-05 15:00:33

鴻蒙HarmonyOS應用

2022-04-13 11:12:43

鴻蒙輕內核信號量模塊操作系統

2021-05-25 09:28:34

鴻蒙HarmonyOS應用

2021-10-20 16:08:57

鴻蒙HarmonyOS應用

2021-12-01 15:59:22

鴻蒙HarmonyOS應用

2021-05-31 20:30:55

鴻蒙HarmonyOS應用

2022-03-31 16:26:49

鴻蒙源碼分析進程管理
點贊
收藏

51CTO技術棧公眾號

国产一区在线精品| 欧美亚洲综合视频| 私拍精品福利视频在线一区| 精品国产91久久久| 国产精品刘玥久久一区| 一区二区三区国产在线观看| 日韩国产伦理| 日本一道高清一区二区三区| 欧美日韩国产精品成人| 日韩中文字幕免费在线| 欧美一级二区| 亚洲网友自拍| 亚洲精品午夜精品| 国产欧美123| 男女羞羞视频在线观看| 精品中文av资源站在线观看| 日韩三级视频在线观看| 精品免费二区三区三区高中清不卡 | 国产精品 日产精品 欧美精品| 欧美日韩伦理| 日韩在线欧美在线国产在线| 免费的黄网站在线观看| 一区二区欧美视频| 毛片毛片毛片毛片毛片毛片毛片毛片毛片 | 日韩电影天堂视频一区二区| 日韩一区二区在线| 九色成人免费视频| 中老年在线免费视频| 在线影视一区二区三区| 美女露隐私免费网站| 久久久久久久免费视频了| 一区二区三区四区视频在线| 欧美午夜不卡| 91久久国产自产拍夜夜嗨| 欧美亚洲国产激情| 欧美一区深夜视频| 成人在线视频你懂的| 久久精品免费播放| 91精品福利观看| 久久av在线看| 白白在线精品| 欧洲永久精品大片ww免费漫画| 岛国av一区| 97精品一区二区视频在线观看| 亚洲网站免费| 美乳少妇欧美精品| 国产午夜精品一区在线观看| xxx欧美精品| 中文字幕区一区二区三| 欧美疯狂性受xxxxx另类| 91精品国产自产观看在线| 日韩最新在线视频| 日本99精品| 91精品国产高清自在线| 日韩av网址大全| 国产在线精品一区免费香蕉| 91成人网在线观看| 美女视频久久| 久久国内精品视频| 国产黄页在线观看| 亚洲特黄一级片| 四虎成人免费在线| 制服丝袜av成人在线看| 精精国产xxxx视频在线野外| 正在播放国产一区| 欧美a大片欧美片| 91在线精品视频| 日本美女一区二区三区视频| 成人国产在线看| 国产精品九色蝌蚪自拍| 国产在线制服美女| 91精品国产一区二区三区| 成人性生交大片免费观看网站| 欧美激情第1页| 自拍欧美日韩| 四虎免费在线观看视频| 国产亚洲精品bt天堂精选| 中文字幕一区二区三区免费视频| 91精品欧美福利在线观看| 小视频免费在线观看| 欧美成人sm免费视频| 日本一区二区三区视频| 成人自拍视频网| 丝袜a∨在线一区二区三区不卡| 日本日本精品二区免费| 国产精品自拍一区| 综合激情久久| xxxxxx在线观看| 蜜桃精品在线观看| 91.com在线| 国产精品高潮呻吟| eeuss影院www在线播放| 亚洲免费电影在线观看| 日韩激情毛片| 五月天亚洲综合情| 国产精品人妖ts系列视频| av在线电影免费观看| 色妞色视频一区二区三区四区| 成人精品电影| 中国一级大黄大黄大色毛片| 亚洲日本一区二区三区| eeuss鲁一区二区三区| 国产69精品久久久久久| 视频一区视频二区中文| 91亚洲免费视频| 日韩三级视频在线看| 中文字幕亚洲在线观看| 国产高清在线精品一区二区三区| 国产在线看一区| 精东传媒在线观看| 亚洲视频第一页| 欧美不卡高清| 91人人澡人人爽人人精品| 欧美一区二区三区四区久久| 国产极品模特精品一二| 亚洲精品视频一二三| 国产精品电影院| 久久香蕉一区| 国产精品天天狠天天看| 国产成人午夜精品5599| 精品美女视频在线观看免费软件 | 男人在线资源站| 国内外成人免费激情在线视频网站| 国产精品亚洲产品| 2019一级黄色毛片免费看网| 日韩成人性视频| 99久久综合| 亚洲少妇第一页| 亚洲人成电影网站色xx| 亚洲天堂黄色| 在线观看成年人视频| 欧美成人精品在线| 国产一区二区免费在线| 国产网友自拍视频导航网站在线观看| 日本高清久久天堂| 久久综合九色综合欧美就去吻| 国产理论电影在线| 国产精品污www一区二区三区| 亚洲免费视频中文字幕| 黑人一区二区三区| 在线免费观看成人| 884aa四虎影成人精品一区| 久久国产精品亚洲人一区二区三区 | 成人欧美一区二区三区小说 | 亚洲婷婷综合色高清在线| av日韩电影| 欧美在线日韩精品| 欧洲国产伦久久久久久久| 国产成人精品在线视频| 亚洲专区国产精品| 韩国精品久久久| 波多野结衣在线观看| 久久99精品久久久久子伦| 欧美视频在线观看 亚洲欧| 亚洲自拍电影| 国产夫妻在线视频| 亚洲**2019国产| 国产欧美精品一区二区色综合| 成人久久网站| 黄色一级在线视频| 中文字幕日韩高清| 成人精品gif动图一区| 久久爱91午夜羞羞| 女同性恋一区二区| 正在播放欧美视频| 久久久久久久久久电影| 伊人精品综合| 福利片免费在线观看| 日本亚洲欧美成人| 亚洲一区影音先锋| 99久久精品网| av电影在线观看| 免费成人深夜夜行视频| 精品乱码亚洲一区二区不卡| 日本不卡视频在线观看| 中文不卡1区2区3区| 国产成a人亚洲精v品在线观看| 亚洲性69xxxbbb| 91麻豆免费视频| 亚洲人成精品久久久| 男人的天堂在线视频| 日本一区二区不卡高清更新| 亚洲第一区第二区| 99久免费精品视频在线观看| 国产精品超碰| 青青草免费在线| 天堂一区二区三区 | 蜜桃视频www网站在线观看| 麻豆md0077饥渴少妇| 精品国产欧美一区二区五十路 | 久久中文字幕在线| 国产精品日韩成人| 99国内精品久久久久久久| 亚洲wwwww| 欧美亚洲另类色图| 清纯唯美亚洲综合| 7777精品伊人久久久大香线蕉完整版 | 国产精品私人影院| 66国产精品| 中文在线资源|