Каков наилучший способ преобразования сетевого bitcount в сетевую маску?

StackOverflow https://stackoverflow.com/questions/218604

  •  03-07-2019
  •  | 
  •  

Вопрос

Например, если у меня есть спецификация сети, например 172.20.10.0 / 24, "24" - это количество битов.Каков наилучший способ преобразовать это в сетевую маску типа 0xffffff00?

Это было полезно?

Решение

Зачем тратить время на вычитания или троичные выражения?

int suffix = 24;
int mask = 0xffffffff ^ 0xffffffff >> suffix;

Если вы знаете, что ваше целое число имеет длину ровно 32 бита, вам нужно набрать 0xffffffff только один раз.

int32_t mask = ~(0xffffffff >> suffix);

Оба компилируются с одинаковым кодом сборки.

Другие советы

Предполагая 32-битную маску и 32-битное int.

int keepBits = 24;  /* actually get it from somewhere else? */

int mask = (0xffffffff >> (32 - keepBits )) << (32 - keepBits);

Примечание. Это не обязательно ответ на вопрос "Какой лучший способ получить маску сети для интерфейса?"

Я всегда так делаю (в вашем случае cidr = 24):

uint32_t ipv4Netmask;

ipv4Netmask = 0xFFFFFFFF;
ipv4Netmask <<= 32 - cidr;
ipv4Netmask = htonl(ipv4Netmask);

Это будет работать только с ipv4Netmask, чтобы быть на самом деле uint32_t, не делайте его int, так как int не должен быть 32-битным в каждой системе. Результат преобразуется в сетевой порядок байтов, как и ожидают большинство системных функций.

Обратите внимание, что этот код завершится ошибкой, если cidr равен нулю, так как тогда код сместит 32-битную переменную на 32 бита, и, хотите верьте, хотите нет, это неопределенное поведение в C. Можно ожидать результат всегда будет нулевым, но стандарт Си говорит, что это не определено с самого начала. Если ваш CIDR может быть нулевым (что будет разрешено только в любом заполнителе IP-адреса 0.0.0.0/0), тогда код должен перехватывать особый случай.

Это не вопрос программирования, но в Linux вы можете использовать whatmask.

whatmask 72.20.10.0/24

возвращает

IP Entered = ..................: 72.20.10.0
CIDR = ........................: /24
Netmask = .....................: 255.255.255.0
Netmask (hex) = ...............: 0xffffff00
Wildcard Bits = ...............: 0.0.0.255
------------------------------------------------
Network Address = .............: 72.20.10.0
Broadcast Address = ...........: 72.20.10.255
Usable IP Addresses = .........: 254
First Usable IP Address = .....: 72.20.10.1
Last Usable IP Address = ......: 72.20.10.254
int keepbits = 24;
int mask = keepbits > 0 ? 0x00 - (1<<(32 - keepbits)) : 0xFFFFFFFF;

Вот решение в VBScript, FWIW

option explicit

'whatmask 72.20.10.0/24
If WScript.Arguments.Unnamed.Count < 1 Then
    WScript.Echo "WhatMask xxx.xxx.xxx.xxx/xx"
    Wscript.Quit
End If

Dim sToFind
Dim aParts
Dim nSubnet

sToFind = WScript.Arguments(0)
aParts = Split( sToFind, "/", 2 )
nSubnet = aParts(1)

if nSubnet < 1 or nSubnet > 32 then
  WScript.echo "Subnet out of range [1..32]"
  Wscript.quit
end if

Dim sBinary
sBinary = String( nSubnet, "1")
sBinary = sBinary & String( 32 - nSubnet, "0" )

wscript.echo "0x" & lcase( binary2hexadecimal( sBinary ) )

function binary2hexadecimal( sBin )
  dim sSlice
  dim sResult
  dim i
  for i = 1 to len( sBin ) step 4
    sSlice = mid( sBin, i, 4 )
    sResult = sResult & hex( binary2decimal( sSlice ) )
  next
  binary2hexadecimal = sResult
end function

function binary2decimal( sFourbits )
  dim i
  dim bit
  dim nResult
  nResult = 0
  for i = 4 to 1 step -1
    bit = mid(sFourbits, i, 1 )
    nResult = nResult * 2 + bit
  next
  binary2decimal = nResult
end function

Из командной строки

>whatmask.vbs 123.12.123.17/23
 0xfffff700

Будьте осторожны, когда вы используете предыдущие ответы с таким кодом, как:

0xFFFFFFFF << 32 - cidr

или

-1 << 32 - cidr

По крайней мере, в C # сначала это замаскирует количество сдвигов значением 0x1F.Итак, для cidr с префиксом 0 (то есть всего диапазона адресов IPv4):

int cidr=0;
0xFFFFFFFF << (32 - cidr) == 0xFFFFFFFF

а это совсем не то, чего ты хочешь.Вместо этого вам следует использовать:

int cidr=0;
(int)(0xFFFFFFFFL << (32 - cidr)) == 0

Вы можете попробовать что-нибудь простое, например, взять число битов и разделить их на 4. Это даст вам первые F в маске. А затем возьмите остаток и переключитесь с 0 бит на 3 бита.

/* C# version merging some of the other contributions and corrected for byte order. */

int cidr = 24;

var ipv4Netmask = 0xFFFFFFFF;

ipv4Netmask <<= 32 - cidr;

byte[] bytes = BitConverter.GetBytes(ipv4Netmask);

Array.Reverse(bytes);

ipv4Netmask = BitConverter.ToUInt32(bytes, 0);    

// mask is now ready for use such as:

var netmask = new IPAddress(ipv4Netmask);
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top