Linux中常見的ip地址以及socket 地址表示方法

最近與IP地址以及socket 地址打交道比較多, 碰到幾個常見的套接字:

  1. sockaddr_in - ipv4 地址, number表示法 4個字節(jié)即可, 比如字面量表示法是"255.255.255.255", 轉(zhuǎn)成數(shù)字表示法, ff.ff.ff.ff => ffffffff
  2. sock_addr_in6 -- ipv6地址, number表示法 16個字節(jié)即可
  3. sockaddr -- 常見socket API的 socket 地址通用表示法, 16個字節(jié)
  4. sockaddr_storage -- 為兼容ipv6的socket地址通用表示法, 28 個字節(jié)

IPv4 address以及 Socket address for IPv4 , in_addr 是4個字節(jié), IPv4 的Socket address是16個字節(jié)

typedef __uint8_t       sa_family_t;
typedef __uint16_t      in_port_t;
typedef __uint32_t      in_addr_t;      /* base type for internet address */

/*
 * Internet address (a structure for historical reasons)
 * Ipv4 的地址, 歷史原因
 */
struct in_addr {
    in_addr_t s_addr;
};

/*
 * Socket address, internet style.
 */
struct sockaddr_in {
    __uint8_t       sin_len;
    sa_family_t     sin_family;
    in_port_t       sin_port;
    struct  in_addr sin_addr;
    char            sin_zero[8];
};

IPv6 address以及 Socket address for IPv6 , IPv6 的Socket address 是28個字節(jié)

typedef __uint8_t       sa_family_t;
typedef __uint16_t      in_port_t;

/*
 * IPv6 address 16個字節(jié) 128bit
 */
typedef struct in6_addr {
    union {
        __uint8_t   __u6_addr8[16];
        __uint16_t  __u6_addr16[8];
        __uint32_t  __u6_addr32[4];
    } __u6_addr;                    /* 128-bit IP6 address */
} in6_addr_t;

#define s6_addr   __u6_addr.__u6_addr8

#define INET6_ADDRSTRLEN 46

/*
 * Socket address for IPv6
 */
struct sockaddr_in6 {
    __uint8_t       sin6_len;       /* length of this struct(sa_family_t) */
    sa_family_t     sin6_family;    /* AF_INET6 (sa_family_t) */
    in_port_t       sin6_port;      /* Transport layer port # (in_port_t) */
    __uint32_t      sin6_flowinfo;  /* IP6 flow information */
    struct in6_addr sin6_addr;      /* IP6 address */
    __uint32_t      sin6_scope_id;  /* scope zone index */
};

為了統(tǒng)一各種協(xié)議, socket對應(yīng)的接口就定義了兩個通用結(jié)構(gòu), 分別是sockaddr(16字節(jié))sockaddr_storage(128字節(jié)),兩個通用套接字的結(jié)構(gòu)體, 其中sockaddr_storage是為了適配sockaddr_in6(128字節(jié))這樣長度比較大的協(xié)議而后來定義的闽颇,如果需要用到sockaddr_storage這樣的通用套接字我擂,則強轉(zhuǎn)為sockaddr,并且長度用sizeof(struct sockaddr_storage)

下圖中, struct sockaddr 是16個字節(jié), struct sockaddr_storage是28個字節(jié)

/*
 * [XSI] Structure used by kernel to store most addresses.
 */
struct sockaddr {
    __uint8_t       sa_len;         /* total length */
    sa_family_t     sa_family;      /* [XSI] address family */
    char            sa_data[14];    /* [XSI] addr value (actually larger) */
};

/*
 * RFC 2553: protocol-independent placeholder for socket addresses
 */
#define _SS_MAXSIZE     128
#define _SS_ALIGNSIZE   (sizeof(__int64_t))
#define _SS_PAD1SIZE    \
            (_SS_ALIGNSIZE - sizeof(__uint8_t) - sizeof(sa_family_t))
#define _SS_PAD2SIZE    \
            (_SS_MAXSIZE - sizeof(__uint8_t) - sizeof(sa_family_t) - \
                            _SS_PAD1SIZE - _SS_ALIGNSIZE)

/*
 * [XSI] sockaddr_storage
 */
struct sockaddr_storage {
    __uint8_t       ss_len;                             /* address length */
    sa_family_t     ss_family;                          /* [XSI] address family */
    char            __ss_pad1[_SS_PAD1SIZE];
    __int64_t       __ss_align;                         /* force structure storage alignment */
    char            __ss_pad2[_SS_PAD2SIZE];
};

在Linux 網(wǎng)絡(luò)編程中, 不管用的是什么協(xié)議, 使用的都是同一套socket API, 而每個協(xié)議表示網(wǎng)絡(luò)地址的結(jié)構(gòu)體是不一樣的, 如IPv4是struct sockaddr_in, IPv6是struct sockaddr_in6, 所以需要將所有這些表示網(wǎng)絡(luò)地址的結(jié)構(gòu)體抽象成一個通用的結(jié)構(gòu)體, 抽象之后的結(jié)構(gòu)體便是struct sockaddr.

常見的涉及通用socket結(jié)構(gòu)體的API為:

/*application->kernel*/
int bind   (int sockfd, struct sockaddr *my_addr, socklen_t addrlen);

int connect(int sockfd, const struct sockaddr *serv_addr,
            socklen_t addrlen);

int sendto (int s, const void *msg, size_t len, int flags,
            const struct sockaddr *to, socklen_t tolen);

/*kernel->application*/
int  accept     (int s, struct sockaddr *addr, socklen_t *addrlen);

int  recvfrom   (int s,  void  *buf, size_t len, int flags,
                 struct sockaddr *from, socklen_t *fromlen);

int  getpeername(int s, struct sockaddr *name, socklen_t *namelen);
int  getsockname(int s, struct sockaddr *name, socklen_t *namelen);

那為什么會出現(xiàn)struct sockaddr_storage這個通用結(jié)構(gòu)體呢齐蔽?

身為一個通用的地址數(shù)據(jù)結(jié)構(gòu), 它的大小得要是所有具體協(xié)議地址結(jié)構(gòu)大小的最大值, 可是sizeof(struct sockaddr) = 16, 而sizeof(struct sockaddr_in6) = 28, 顯然struct sockaddr這個通用的數(shù)據(jù)結(jié)構(gòu)hold不住IPv6啊,所以struct sockaddr_storage這個新結(jié)構(gòu)橫空出世了舆驶,它的大小為128字節(jié)拓轻,應(yīng)該能裝得下目前所以協(xié)議的地址結(jié)構(gòu)了。

在涉及多種協(xié)議的網(wǎng)絡(luò)編程中批幌,用struct sockaddr_storage這個結(jié)構(gòu)體來表示通信地址础锐,調(diào)用socket API時,需要將struct sockaddr_storage強制類型轉(zhuǎn)換為struct sockaddr荧缘,地址長度為sizeof(struct sockaddr_storage)皆警。如:

struct sockaddr_storage addr;
/*
地址賦值
*/
sendto (s, *msg, len, flags, (struct sockaddr *)&addr, sizeof(struct sockaddr_storage));

一個IPv4/IPv6混合編程的Demo:

struct sockaddr_storage addr;
memset(&addr, 0, sizeof(struct sockaddr_storage));
if (isIPv6 == TRUE)
{
    struct sockaddr_in6 *addr_v6 = (struct sockaddr_in6 *)&addr;
    addr_v6->sin6_family = AF_INET6;
    addr_v6->sin6_port = 1234;
    inet_pton(AF_INET6, “2001:3211::1”, &(addr_v6->sin6_addr));
}
else
{
    struct sockaddr_in *addr_v4 = (struct sockaddr_in *)&addr;
    addr_v4->sin_family = AF_INET;
    addr_v4->sin_port = 1234;
    inet_aton(“192.168.1.228”, &(addr_v4->sin_addr));
}

sendto(sock, buf, len, 0, (struct sockaddr *)&addr, sizeof(struct sockaddr_storage));

參考文獻

https://www.cnblogs.com/tanghuimin0713/p/3425936.html

https://blog.csdn.net/weixin_44874963/article/details/89395292

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市截粗,隨后出現(xiàn)的幾起案子耀怜,更是在濱河造成了極大的恐慌,老刑警劉巖桐愉,帶你破解...
    沈念sama閱讀 219,188評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件财破,死亡現(xiàn)場離奇詭異,居然都是意外死亡从诲,警方通過查閱死者的電腦和手機左痢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人俊性,你說我怎么就攤上這事略步。” “怎么了定页?”我有些...
    開封第一講書人閱讀 165,562評論 0 356
  • 文/不壞的土叔 我叫張陵趟薄,是天一觀的道長。 經(jīng)常有香客問我典徊,道長杭煎,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,893評論 1 295
  • 正文 為了忘掉前任卒落,我火速辦了婚禮羡铲,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好统舀,可當(dāng)我...
    茶點故事閱讀 67,917評論 6 392
  • 文/花漫 我一把揭開白布恒削。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,708評論 1 305
  • 那天褂萧,我揣著相機與錄音,去河邊找鬼葵萎。 笑死导犹,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的羡忘。 我是一名探鬼主播谎痢,決...
    沈念sama閱讀 40,430評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼卷雕!你這毒婦竟也來了节猿?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,342評論 0 276
  • 序言:老撾萬榮一對情侶失蹤漫雕,失蹤者是張志新(化名)和其女友劉穎滨嘱,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體浸间,經(jīng)...
    沈念sama閱讀 45,801評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡太雨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,976評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了魁蒜。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片囊扳。...
    茶點故事閱讀 40,115評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡吩翻,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出锥咸,到底是詐尸還是另有隱情狭瞎,我是刑警寧澤,帶...
    沈念sama閱讀 35,804評論 5 346
  • 正文 年R本政府宣布搏予,位于F島的核電站熊锭,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏雪侥。R本人自食惡果不足惜碗殷,卻給世界環(huán)境...
    茶點故事閱讀 41,458評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望校镐。 院中可真熱鬧,春花似錦捺典、人聲如沸鸟廓。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽引谜。三九已至,卻和暖如春擎浴,著一層夾襖步出監(jiān)牢的瞬間员咽,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評論 1 272
  • 我被黑心中介騙來泰國打工贮预, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留贝室,地道東北人。 一個月前我還...
    沈念sama閱讀 48,365評論 3 373
  • 正文 我出身青樓仿吞,卻偏偏與公主長得像滑频,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子唤冈,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,055評論 2 355

推薦閱讀更多精彩內(nèi)容