1. 程式人生 > >VPP程式碼閱讀中文註解--vec_bootstrap.h

VPP程式碼閱讀中文註解--vec_bootstrap.h

/** \file
    Vector bootstrap header file
*/

/* Bootstrap include so that #include <vppinfra/mem.h> can include e.g.
   <vppinfra/mheap.h> which depends on <vppinfra/vec.h>. */

/** \brief vector header structure

   Bookkeeping header preceding vector elements in memory.
   User header information may preceed standard vec header.
   If you change u32 len -> u64 len, single vectors can
   exceed 2**32 elements. Clib heaps are vectors. */

typedef struct
{
#if CLIB_VEC64 > 0
  u64 len;
#else
  u32 len; /**< Number of elements in vector (NOT its allocated length). */
  u32 dlmalloc_header_offset;	/**< offset to memory allocator offset  */
#endif
  u8 vector_data[0];  /**< Vector data . */
} vec_header_t;

向量即動態陣列,這裡的vec_header_t只是動態陣列的頭結構,表示這個陣列有多少個數組元素,即它的規模。陣列0號元素的位置是vector_data對應的記憶體地址。

使用這個結構的時候,也可以在vec_header_t前再加1個額外的頭結構,描述這個具體動態陣列整體更詳細的一些資訊,這個具體使用時具體分析。

dlmalloc_header_offset暫不分析。

/** \brief Find the vector header

    Given the user's pointer to a vector, find the corresponding
    vector header

    @param v pointer to a vector
    @return pointer to the vector's vector_header_t
*/
#define _vec_find(v)	((vec_header_t *) (v) - 1)

根據動態陣列0號元素的地址計算出vec_header_t頭部的地址

#define _vec_round_size(s) \
  (((s) + sizeof (uword) - 1) &~ (sizeof (uword) - 1))

可能增加s的值,使得其是cpu字長的整數倍。如4的整數倍或者8的整數倍

 

always_inline uword
vec_header_bytes (uword header_bytes)
{
  return round_pow2 (header_bytes + sizeof (vec_header_t),
		     sizeof (vec_header_t));
}

使用者定義的頭部長度加上動態陣列自己的頭部長度,並8位元組對齊。

使用者定義的頭部長度可以為0。

/** \brief Find a user vector header

    Finds the user header of a vector with unspecified alignment given
    the user pointer to the vector.
*/

always_inline void *
vec_header (void *v, uword header_bytes)
{
  return v - vec_header_bytes (header_bytes);
}

根據動態陣列第0號元素的地址獲取使用者自定義頭的記憶體地址,如果沒有使用者自定義頭,則獲取到的是動態陣列頭的地址。

 

/** \brief Find the end of user vector header

    Finds the end of the user header of a vector with unspecified
    alignment given the user pointer to the vector.
*/

always_inline void *
vec_header_end (void *v, uword header_bytes)
{
  return v + vec_header_bytes (header_bytes);
}

根據使用者自定義頭的記憶體地址,獲取動態陣列第0號元素的地址

 

always_inline uword
vec_aligned_header_bytes (uword header_bytes, uword align)
{
  return round_pow2 (header_bytes + sizeof (vec_header_t), align);
}

always_inline void *
vec_aligned_header (void *v, uword header_bytes, uword align)
{
  return v - vec_aligned_header_bytes (header_bytes, align);
}

always_inline void *
vec_aligned_header_end (void *v, uword header_bytes, uword align)
{
  return v + vec_aligned_header_bytes (header_bytes, align);
}

和前面的程式碼類似,只是不再是8位元組記憶體對齊,而是通過align引數來控制對齊。

 

/** \brief Number of elements in vector (lvalue-capable)

   _vec_len (v) does not check for null, but can be used as a lvalue
   (e.g. _vec_len (v) = 99).
*/

#define _vec_len(v)	(_vec_find(v)->len)

讀取動態陣列的元素個數,可以作為左值,進行賦值

 

/** \brief Number of elements in vector (rvalue-only, NULL tolerant)

    vec_len (v) checks for NULL, but cannot be used as an lvalue.
    If in doubt, use vec_len...
*/

#define vec_len(v)	((v) ? _vec_len(v) : 0)

同上,只能做為右值,但右空指標判斷。

如果v為空指標,表示當前動態陣列沒有有效的元素,可能只有陣列頭部,也可能有使用者自定義的頭部,還可能動態陣列的記憶體空間已經申請好,只是沒有使用而已。

 

/** \brief Reset vector length to zero
    NULL-pointer tolerant
*/

#define vec_reset_length(v) do { if (v) _vec_len (v) = 0; } while (0)

將陣列長度置0,其實就是宣佈當前陣列中的內容失效。

 

/** \brief Number of data bytes in vector. */

#define vec_bytes(v) (vec_len (v) * sizeof (v[0]))

計算當前陣列資料部分已佔用的記憶體空間(位元組數). 即每一個數組元素的大小乘以陣列元素個數

 

/** \brief Total number of bytes that can fit in vector with current allocation. */

#define vec_capacity(v,b)							\
({										\
  void * _vec_capacity_v = (void *) (v);					\
  uword _vec_capacity_b = (b);							\
  _vec_capacity_b = sizeof (vec_header_t) + _vec_round_size (_vec_capacity_b);	\
  _vec_capacity_v ? clib_mem_size (_vec_capacity_v - _vec_capacity_b) : 0;	\
})

計算動態陣列佔用的記憶體空間,也包含2種類型的頭部,不僅僅是資料部分

/** \brief Total number of elements that can fit into vector. */
#define vec_max_len(v) (vec_capacity(v,0) / sizeof (v[0]))

當前已分配的記憶體最多容納的元素個數,實際上達不到,因為忽略了8位元組的陣列頭部。

 

/** \brief End (last data address) of vector. */
#define vec_end(v)	((v) + vec_len (v))

當前陣列元素的最後一個元素的下一個位置,新元素在這裡追加

 

/** \brief True if given pointer is within given vector. */
#define vec_is_member(v,e) ((e) >= (v) && (e) < vec_end (v))

判斷一個元素是否合法的陣列元素,是否越界

 

/** \brief Get vector value at index i checking that i is in bounds. */
#define vec_elt_at_index(v,i)			\
({						\
  ASSERT ((i) < vec_len (v));			\
  (v) + (i);					\
})

讀取第i號陣列元素

 

/** \brief Get vector value at index i */
#define vec_elt(v,i) (vec_elt_at_index(v,i))[0]

同上,不過不是指標形式,而是結構體形式。

/** \brief Vector iterator */
#define vec_foreach(var,vec) for (var = (vec); var < vec_end (vec); var++)

/** \brief Vector iterator (reverse) */
#define vec_foreach_backwards(var,vec) \
for (var = vec_end (vec) - 1; var >= (vec); var--)

/** \brief Iterate over vector indices. */
#define vec_foreach_index(var,v) for ((var) = 0; (var) < vec_len (v); (var)++)

/** \brief Iterate over vector indices (reverse). */
#define vec_foreach_index_backwards(var,v) \
  for ((var) = vec_len((v)) - 1; (var) >= 0; (var)--)

以指標或者下標形式遍歷動態陣列中的每一個元素