Получение и настройка шлюза по умолчанию в C с поддержкой Windows XP - PullRequest
0 голосов
/ 05 марта 2011

Я нашел интерфейсы IP Helper для управления информацией, связанной с ip, но я не нашел способа получить и / или изменить адрес шлюза для данного адаптера, используя интерфейсы, поддерживаемые в Windows XP. Есть идеи?

Ответы [ 4 ]

2 голосов
/ 05 марта 2011

Если вы можете использовать WMI, существует класс Win32_NetworkAdapterConfiguration .Если вам нужны простые старые API C, я нашел это («Настройка адреса шлюза по умолчанию» внизу).

1 голос
/ 07 марта 2011

Функции GetAdaptersInfo() и GetAdaptersAddresses() возвращают информацию о шлюзе для каждого адаптера.

0 голосов
/ 10 октября 2017
    public enum ForwardType
    {
        Other = 1,
        Invalid = 2,
        Direct = 3,
        Indirect = 4
    }

    public enum ForwardProtocol
    {
        Other = 1,
        Local = 2,
        NetMGMT = 3,
        ICMP = 4,
        EGP = 5,
        GGP = 6,
        Hello = 7,
        RIP = 8,
        IS_IS = 9,
        ES_IS = 10,
        CISCO = 11,
        BBN = 12,
        OSPF = 13,
        BGP = 14,
        NT_AUTOSTATIC = 10002,
        NT_STATIC = 10006,
        NT_STATIC_NON_DOD = 10007
    }

    [StructLayout(LayoutKind.Sequential)]
    public struct MIB_IPFORWARDROW
    {
        public uint dwForwardDest;
        public uint dwForwardMask;
        public int dwForwardPolicy;
        public uint dwForwardNextHop;
        public int dwForwardIfIndex;
        public ForwardType dwForwardType;
        public ForwardProtocol dwForwardProto;
        public int dwForwardAge;
        public int dwForwardNextHopAS;
        public int dwForwardMetric1;
        public int dwForwardMetric2;
        public int dwForwardMetric3;
        public int dwForwardMetric4;
        public int dwForwardMetric5;

        public static List<MIB_IPFORWARDROW> FromByteArray(byte[] buffer)
        {
            List<MIB_IPFORWARDROW> ret = new List<MIB_IPFORWARDROW>();

            int n =
                (buffer[3] << 24) +
                (buffer[2] << 16) +
                (buffer[1] << 8) +
                (buffer[0] << 0);

            int offset = sizeof(int);
            for (int i = 0; i < n; i++)
            {
                MIB_IPFORWARDROW map = new MIB_IPFORWARDROW();
                IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(map));
                Marshal.StructureToPtr(map, ptr, false);
                Marshal.Copy(buffer, offset, ptr, Marshal.SizeOf(map));
                map = (MIB_IPFORWARDROW)Marshal.PtrToStructure(ptr, typeof(MIB_IPFORWARDROW));
                Marshal.FreeHGlobal(ptr);
                ret.Add(map);
                offset += Marshal.SizeOf(map);
            }
            return ret;
        }
    }

    [DllImport("Iphlpapi.dll", CharSet = CharSet.Auto)]
    public static extern int GetIpForwardTable(byte[] pIpForwardTable, ref ulong pdwSize, bool bOrder);

    [DllImport("Iphlpapi.dll", CharSet = CharSet.Auto)]
    public static extern int DeleteIpForwardEntry(ref MIB_IPFORWARDROW pRoute);

    [DllImport("Iphlpapi.dll", CharSet = CharSet.Auto)]
    public static extern int SetIpForwardEntry(ref MIB_IPFORWARDROW pRoute);

    private static uint IPToUIntR(IPAddress ip)
    {
        uint ret = 0;
        byte[] ipBytes = ip.GetAddressBytes();

        ret += (uint)ipBytes[0] << 24;
        ret += (uint)ipBytes[1] << 16;
        ret += (uint)ipBytes[2] << 8;
        ret += (uint)ipBytes[3];

        return ret;
    }

    public static List<MIB_IPFORWARDROW> GetIPForwardTable(int index)
    {
        List<MIB_IPFORWARDROW> ret = new List<MIB_IPFORWARDROW>();
        ulong size = 0;
        int err = GetIpForwardTable(null, ref size, false);
        byte[] buffer = new byte[size];
        err = GetIpForwardTable(buffer, ref size, false);

        if (err != 0)
        {
            throw new System.ComponentModel.Win32Exception(err, $"GetIPForwardTable return with error code {err}");
        }

        foreach (MIB_IPFORWARDROW mipr in MIB_IPFORWARDROW.FromByteArray(buffer))
        {
            if (mipr.dwForwardIfIndex != index)
            {
                continue;
            }
            ret.Add(mipr);
        }
        return ret;
    }

    public static void SetDefaultGateway(int index, string gateway)
    {
        int err;
        int i;
        List<MIB_IPFORWARDROW> l = GetIPForwardTable(index);
        MIB_IPFORWARDROW mipr = new MIB_IPFORWARDROW();
        for (i = 0; i < l.Count; ++i)
        {
            // 0.0.0.0 default gateway
            if (l[i].dwForwardDest != 0)
            {
                continue;
            }

            mipr = l[i];
            err = DeleteIpForwardEntry(ref mipr);
            if (err != 0)
            {
                throw new System.ComponentModel.Win32Exception(err, $"DeleteIpForwardEntry return with error code {err}");
            }
            break;
        }
        mipr.dwForwardNextHop = IPToUIntR(IPAddress.Parse(gateway));

        err = SetIpForwardEntry(ref mipr);
        if (err != 0)
        {
            throw new System.ComponentModel.Win32Exception(err, $"DeleteIpForwardEntry return with error code {err}");
        }
    }

В приведенном выше коде используются pinvokes для использования функции iphelper в .net, поэтому просто забудьте о pinvokes и используйте код в качестве ссылки для приложения win32.

https://msdn.microsoft.com/en-us/library/windows/desktop/aa366363(v=vs.85).aspx

редактирование: Вы можете использовать приведенный ниже код для получения информации об адаптере. и используйте поле Index, когда вам нужно использовать SetDefaultGateway.

    const int MAX_ADAPTER_ADDRESS_LENGTH = 8;
    const int MAX_ADAPTER_DESCRIPTION_LENGTH = 128;
    const int MAX_ADAPTER_NAME_LENGTH = 256;
    const int MAX_ADAPTER_NAME = 128;

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public struct IP_ADDRESS_STRING
    {
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
        public string Address;
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public struct IP_ADDR_STRING
    {
        public IntPtr Next;
        public IP_ADDRESS_STRING IpAddress;
        public IP_ADDRESS_STRING IpMask;
        public ulong Context;
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public struct IP_ADAPTER_INFO
    {
        public IntPtr Next;
        public Int32 ComboIndex;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = MAX_ADAPTER_NAME_LENGTH + 4)]
        public string AdapterName;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = MAX_ADAPTER_DESCRIPTION_LENGTH + 4)]
        public string AdapterDescription;
        public UInt32 AddressLength;
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = MAX_ADAPTER_ADDRESS_LENGTH)]
        public byte[] Address;
        public Int32 Index;
        public UInt32 Type;
        public UInt32 DhcpEnabled;
        public IntPtr CurrentIpAddress;
        public IP_ADDR_STRING IpAddressList;
        public IP_ADDR_STRING GatewayList;
        public IP_ADDR_STRING DhcpServer;
        public bool HaveWins;
        public IP_ADDR_STRING PrimaryWinsServer;
        public IP_ADDR_STRING SecondaryWinsServer;
        public Int32 LeaseObtained;
        public Int32 LeaseExpires;
    }

    [DllImport("iphlpapi.dll", CharSet = CharSet.Ansi)]
    static extern int GetAdaptersInfo(byte[] pAdapterInfo, ref ulong pBufOutLen);

    public static List<IP_ADAPTER_INFO> GetAllAdapters()
    {
        List<IP_ADAPTER_INFO> ret = new List<IP_ADAPTER_INFO>();

        ulong size = 0;
        int err = GetAdaptersInfo(null, ref size);
        byte[] buffer = new byte[size];
        err = GetAdaptersInfo(buffer, ref size);

        if (err != 0)
        {
            throw new System.ComponentModel.Win32Exception(err, $"GetInterfaceInfo return with error code {err}");
        }

        GCHandle gc = GCHandle.Alloc(buffer, GCHandleType.Pinned);
        IntPtr pEntry = gc.AddrOfPinnedObject();
        do
        {
            IP_ADAPTER_INFO entry = (IP_ADAPTER_INFO)Marshal.PtrToStructure(pEntry, typeof(IP_ADAPTER_INFO));

            ret.Add(entry);

            pEntry = entry.Next;
        }
        while (pEntry != IntPtr.Zero);

        gc.Free();

        return ret;
    }
0 голосов
/ 25 января 2016

Примечание: Win32_NetworkAdapterConfiguration :: SetGateways () установит шлюзы, которые вы указали в параметрах, где существующие шлюзы будут перезаписаны.

Но: Удаление шлюза (ов) , кажется, не работает при использовании SetGateways () Вызов WMI - в моем случае звонок вернулся с успехом, но шлюз все еще присутствовал. Похоже, что многие видели это (установка строк NULL в StringArray или передача NULL для StringArray не помогла).

В моем случае я начал использовать способ WMI (скорее всего, из-за EnableDHCP ()), но также мне нужно было реализовать способ WINAPI, предложенный Люком, чтобы избавиться от аварии с помощью шлюзов. Большое спасибо Люку !!

...