linux内核字节序转换和大小端

作者

QQ群:852283276
微信:arm80x86
微信公众号:青儿创客基地
B站:主页 https://space.bilibili.com/208826118

软件需要在arm和powerpc上同时运行,所以需要考虑大小端转换,搜索了一下内核如何实现大小端转换的,首先在linux/byteorder/generic.h定义了统一的宏,

#define cpu_to_le64 __cpu_to_le64
#define le64_to_cpu __le64_to_cpu
#define cpu_to_le32 __cpu_to_le32
#define le32_to_cpu __le32_to_cpu
#define cpu_to_le16 __cpu_to_le16
#define le16_to_cpu __le16_to_cpu
#define cpu_to_be64 __cpu_to_be64
#define be64_to_cpu __be64_to_cpu
#define cpu_to_be32 __cpu_to_be32
#define be32_to_cpu __be32_to_cpu
#define cpu_to_be16 __cpu_to_be16
#define be16_to_cpu __be16_to_cpu
#define cpu_to_le64p __cpu_to_le64p
#define le64_to_cpup __le64_to_cpup
#define cpu_to_le32p __cpu_to_le32p
#define le32_to_cpup __le32_to_cpup
#define cpu_to_le16p __cpu_to_le16p
#define le16_to_cpup __le16_to_cpup
#define cpu_to_be64p __cpu_to_be64p
#define be64_to_cpup __be64_to_cpup
#define cpu_to_be32p __cpu_to_be32p
#define be32_to_cpup __be32_to_cpup
#define cpu_to_be16p __cpu_to_be16p
#define be16_to_cpup __be16_to_cpup
#define cpu_to_le64s __cpu_to_le64s
#define le64_to_cpus __le64_to_cpus
#define cpu_to_le32s __cpu_to_le32s
#define le32_to_cpus __le32_to_cpus
#define cpu_to_le16s __cpu_to_le16s
#define le16_to_cpus __le16_to_cpus
#define cpu_to_be64s __cpu_to_be64s
#define be64_to_cpus __be64_to_cpus
#define cpu_to_be32s __cpu_to_be32s
#define be32_to_cpus __be32_to_cpus
#define cpu_to_be16s __cpu_to_be16s
#define be16_to_cpus __be16_to_cpus

在内核little_endian.h和big_endian.h中定义了宏,

#include <linux/types.h>
#include <linux/swab.h>

#define __constant_htonl(x) ((__force __be32)___constant_swab32((x)))
#define __constant_ntohl(x) ___constant_swab32((__force __be32)(x))
#define __constant_htons(x) ((__force __be16)___constant_swab16((x)))
#define __constant_ntohs(x) ___constant_swab16((__force __be16)(x))
#define __constant_cpu_to_le64(x) ((__force __le64)(__u64)(x))
#define __constant_le64_to_cpu(x) ((__force __u64)(__le64)(x))
#define __constant_cpu_to_le32(x) ((__force __le32)(__u32)(x))
#define __constant_le32_to_cpu(x) ((__force __u32)(__le32)(x))
#define __constant_cpu_to_le16(x) ((__force __le16)(__u16)(x))
#define __constant_le16_to_cpu(x) ((__force __u16)(__le16)(x))
#define __constant_cpu_to_be64(x) ((__force __be64)___constant_swab64((x)))
#define __constant_be64_to_cpu(x) ___constant_swab64((__force __u64)(__be64)(x))
#define __constant_cpu_to_be32(x) ((__force __be32)___constant_swab32((x)))
#define __constant_be32_to_cpu(x) ___constant_swab32((__force __u32)(__be32)(x))
#define __constant_cpu_to_be16(x) ((__force __be16)___constant_swab16((x)))
#define __constant_be16_to_cpu(x) ___constant_swab16((__force __u16)(__be16)(x))
#define __cpu_to_le64(x) ((__force __le64)(__u64)(x))
#define __le64_to_cpu(x) ((__force __u64)(__le64)(x))
#define __cpu_to_le32(x) ((__force __le32)(__u32)(x))
#define __le32_to_cpu(x) ((__force __u32)(__le32)(x))
#define __cpu_to_le16(x) ((__force __le16)(__u16)(x))
#define __le16_to_cpu(x) ((__force __u16)(__le16)(x))
#define __cpu_to_be64(x) ((__force __be64)__swab64((x)))
#define __be64_to_cpu(x) __swab64((__force __u64)(__be64)(x))
#define __cpu_to_be32(x) ((__force __be32)__swab32((x)))
#define __be32_to_cpu(x) __swab32((__force __u32)(__be32)(x))
#define __cpu_to_be16(x) ((__force __be16)__swab16((x)))
#define __be16_to_cpu(x) __swab16((__force __u16)(__be16)(x))

具体实现则在swab.h中,随便看一个实现,__arch_swab64应该是利用cpu指令集做加速,否则用常规实现,这段代码可以移植到用户态使用。

/*
 * casts are necessary for constants, because we never know how for sure
 * how U/UL/ULL map to __u16, __u32, __u64. At least not in a portable way.
 */
#define ___constant_swab16(x) ((__u16)(				\
	(((__u16)(x) & (__u16)0x00ffU) << 8) |			\
	(((__u16)(x) & (__u16)0xff00U) >> 8)))

#define ___constant_swab32(x) ((__u32)(				\
	(((__u32)(x) & (__u32)0x000000ffUL) << 24) |		\
	(((__u32)(x) & (__u32)0x0000ff00UL) <<  8) |		\
	(((__u32)(x) & (__u32)0x00ff0000UL) >>  8) |		\
	(((__u32)(x) & (__u32)0xff000000UL) >> 24)))

#define ___constant_swab64(x) ((__u64)(				\
	(((__u64)(x) & (__u64)0x00000000000000ffULL) << 56) |	\
	(((__u64)(x) & (__u64)0x000000000000ff00ULL) << 40) |	\
	(((__u64)(x) & (__u64)0x0000000000ff0000ULL) << 24) |	\
	(((__u64)(x) & (__u64)0x00000000ff000000ULL) <<  8) |	\
	(((__u64)(x) & (__u64)0x000000ff00000000ULL) >>  8) |	\
	(((__u64)(x) & (__u64)0x0000ff0000000000ULL) >> 24) |	\
	(((__u64)(x) & (__u64)0x00ff000000000000ULL) >> 40) |	\
	(((__u64)(x) & (__u64)0xff00000000000000ULL) >> 56)))

...

static inline __attribute_const__ __u64 __fswab64(__u64 val)
{
#if defined (__arch_swab64)
	return __arch_swab64(val);
#elif defined(__SWAB_64_THRU_32__)
	__u32 h = val >> 32;
	__u32 l = val & ((1ULL << 32) - 1);
	return (((__u64)__fswab32(l)) << 32) | ((__u64)(__fswab32(h)));
#else
	return ___constant_swab64(val);
#endif
}
  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Linux 内核编程中,经常需要处理网络数据,因此需要进行网络字节序转换。网络字节序是指在网络上传输数据时使用的字节序,它与主机字节序(本地字节序)不同。 在 x86 架构的计算机中,主机字节序是小端序(即低位字节存储在内存的低地址处),而网络字节序是大端序(即高位字节存储在内存的低地址处)。因此,在进行网络数据传输时,需要进行字节序转换Linux 内核提供了一些函数用于进行字节序转换,如: - htons() 和 ntohs() 函数用于将 16 位整数在主机字节序和网络字节序之间转换。 - htonl() 和 ntohl() 函数用于将 32 位整数在主机字节序和网络字节序之间转换。 这些函数都定义在头文件 <arpa/inet.h> 中。例如,将一个 16 位整数从主机字节序转换为网络字节序,可以使用以下代码: ```c #include <arpa/inet.h> uint16_t host_number = 12345; uint16_t network_number = htons(host_number); ``` 同样地,将一个 32 位整数从主机字节序转换为网络字节序,可以使用以下代码: ```c #include <arpa/inet.h> uint32_t host_number = 123456789; uint32_t network_number = htonl(host_number); ``` 反之,将一个从网络中接收到的数据从网络字节序转换为主机字节序,可以使用以下代码: ```c #include <arpa/inet.h> uint16_t network_number = 0x1234; uint16_t host_number = ntohs(network_number); ``` ```c #include <arpa/inet.h> uint32_t network_number = 0x12345678; uint32_t host_number = ntohl(network_number); ``` 需要注意的是,在进行字节序转换时,必须使用适当的数据类型(如 uint16_t 和 uint32_t),否则可能会出现意外的错误。 ### 回答2: 在Linux内核编程中,网络字节序转换是一个常见的操作,因为网络传输中使用的是大端字节序(也称为网络字节序),而计算机内部一般使用的是小端字节序。 在进行网络通信时,需要将数据从主机字节序转换为网络字节序,以确保数据的正确传输和解析。同样,在接收到网络数据后,也需要将数据从网络字节序转换为主机字节序,以便正确地处理和使用这些数据。 在Linux内核中,提供了一系列的函数来进行字节序转换。其中,主要使用的函数为`htonl()`、`htons()`、`ntohl()`和`ntohs()`。它们分别代表主机到网络长整型、主机到网络短整型、网络到主机长整型和网络到主机短整型的转换。 这些函数的命名规则是根据网络字节序的缩写和数据类型的缩写组合而成的。其中,h代表主机字节序(host),n代表网络字节序(network),l代表长整型(long),s代表短整型(short)。 使用这些函数进行字节序转换非常简单。例如,要将一个32位整型数从主机字节序转换为网络字节序,可以使用`htonl()`函数,如下所示: ``` uint32_t value = 12345678; uint32_t network_value = htonl(value); ``` 同样地,要将一个网络字节序的32位整型数转换为主机字节序,可以使用`ntohl()`函数,如下所示: ``` uint32_t network_value = 0x1020304; uint32_t host_value = ntohl(network_value); ``` 总之,在Linux内核编程中,网络字节序转换是非常常见的操作,可以使用`htonl()`、`htons()`、`ntohl()`和`ntohs()`这些函数来实现。这些函数提供了方便、简单和可靠的方式来进行字节序转换,以确保网络数据的正确传输和处理。 ### 回答3: 在Linux内核编程中,网络字节序转换是一个非常重要的概念。网络字节序,也被称为大端字节序,是一种统一的字节序,用于在网络中传输数据。而在计算机内部,一般使用的是主机字节序,也就是与处理器相关的字节序。为了在不同主机之间进行网络通信,就需要进行字节序转换Linux内核提供了一系列函数用于进行字节序转换,这些函数主要集中在`linux/in.h`和`linux/tcp.h`头文件中。其中最常用的函数是`htonl`、`htons`、`ntohl`和`ntohs`。它们分别用于将主机字节序转换为网络字节序和将网络字节序转换为主机字节序。 这些函数的原理是通过位操作和移位运算来实现字节序转换。例如,`htonl`函数可以将32位的整数从主机字节序转换为网络字节序,它将高位字节和低位字节进行位置交换。而`ntohl`函数则在将网络字节序转换为主机字节序时,执行与之相反的操作。 在实际的内核编程中,我们需要根据具体情况选择合适的字节序转换函数。比如,在编写网络设备驱动程序时,需要将设备收到的网络数据转换为主机字节序进行处理,这时就可以使用`ntohl`和`ntohs`函数。而在将数据发送到网络上时,则需要将主机字节序转换为网络字节序,这时可以使用`htonl`和`htons`函数。 总而言之,通过Linux内核提供的网络字节序转换函数,我们可以方便地进行主机字节序和网络字节序之间的转换,从而实现跨主机的网络通信。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值