Home / HomePage / Scripts / ping arp subnet

ping arp subnet


VBscript version

This script performs the painful math to figure out what IP's are included in a subnet range. Optionally, it can ping them all, and perform ARP resolution. Note that ARP resolution only works for those systems that are in the same broadcast domain as the computer where this script runs. ARP doesn't traverse routers.

(grr, I hate having to recreate this page after losing it to the wiki gods, who frown on page renames!)

First, some sample output:

H:\>cscript subnetq.vbs /?
Usage example:

cscript subnetq.vbs /IP:1.2.3.4 /mask:255.255.255.0 /ping

Where:
/IP: specifies a valid IP address and MUST be supplied.
/mask: specifies a valid subnet mask and MUST be supplied
/ping is optional. If supplied, every host in the subnet will be pinged, and result shown.

Most invalid subnet masks will be flagged as such and the script will end.
There is no checking for valid IP address yet!

H:\>cscript subnetq.vbs /ip:10.0.0.2 /mask:255.255.255.240

IP Address: 10.0.0.2
Subnet Mask: 255.255.255.240

Network ID: 10.0.0.0
Broadcast IP = 10.0.0.15
NumberofIPs (including NetworkIP and Broadcast IP): 16
First Host IP = 10.0.0.1
Last Host IP = 10.0.0.14

H:\>cscript subnetq.vbs /ip:10.0.0.2 /mask:255.255.255.240 /ping

IP Address: 10.0.0.2
Subnet Mask: 255.255.255.240

Network ID: 10.0.0.0
Broadcast IP = 10.0.0.15
NumberofIPs (including NetworkIP and Broadcast IP): 16
First Host IP = 10.0.0.1
Last Host IP = 10.0.0.14

Pinging all hosts in subnet.
3 pings per host will be sent; timeout value = 1 second.
If any ping is replied to, the host is considered ALIVE.

10.0.0.1 ping failed, no ARP.
10.0.0.2 is ALIVE, MAC(s) 00-0c-29-73-48-7f
10.0.0.3 ping failed, MAC(s) 00-0c-29-a1-7e-c5 <--this firewalled system doesn't reply to ping. But no firewall fools ARP! 
10.0.0.4 ping failed, no ARP.
10.0.0.5^C
H:\>

 

 

To run this script on your own Windows system, just copy/paste to Notepad. The line numbers will not be part of the copy and they are not needed. Save it as "subnetq.vbs" (or some other name you choose), then open a cmd window, change to the directory you saved in, and run cscript subnetq.vbs - it'll tell you the parameters it needs.

Here is the full script:

  1. Option Explicit
  2.  
  3. dim colArgs, strIPAddress, strSubnetmask
  4. set colArgs = WScript.Arguments.Named
  5. strIPAddress = colArgs.Item("IP")
  6. strSubnetmask = colArgs.Item("mask")
  7. if strIPAddress = "" Then HelpExit
  8. if strSubnetmask = "" Then HelpExit
  9.  
  10. Wscript.echo
  11. Wscript.Echo "IP Address: " & strIPaddress
  12. wscript.echo "Subnet Mask: " & strSubnetmask
  13. Wscript.echo
  14.  
  15. Dim arrNetIDNumIPs
  16. arrNetIDNumIPs = CalcNetIDandNumIPs(strIPaddress, strSubnetmask)
  17. Dim strNetworkID
  18. strNetworkID = arrNetIDNumIPs(0)
  19. Wscript.Echo "Network ID: " & strNetworkID
  20. Dim intNumberofIPs
  21. intNumberofIPs = arrNetIDNumIPs(1)
  22. Dim strBroadcastIP
  23. strBroadcastIP = CalcBroadcastIP(strNetworkID,intNumberofIPs)
  24. wscript.echo "Broadcast IP = " & strBroadcastIP
  25. Wscript.echo "NumberofIPs (including NetworkIP and Broadcast IP): " & intNumberofIPs
  26. Wscript.Echo "First Host IP = " & IncrementIP(strNetworkID)
  27. Wscript.Echo "Last Host IP = " & CalcLastHostIP(strBroadcastIP)
  28.  
  29. If colArgs.Exists("ping") Then Call PingSubnet(strNetworkID, intNumberofIPs)
  30.  
  31. sub PingSubnet(strNetworkID, intNumberofIPs)
  32.         Wscript.Echo
  33.         Wscript.Echo "Pinging all hosts in subnet."
  34.         Wscript.Echo "3 pings per host will be sent; timeout value = 1 second."
  35.         Wscript.Echo "If any ping is replied to, the host is considered ALIVE."
  36.         Wscript.Echo
  37.         Dim strPingIP
  38.         strPingIP = IncrementIP(strNetworkID)
  39.         Call PingHost(strPingIP)
  40.         getArp(strPingIP)
  41.         wscript.echo vbcr
  42.         Dim i
  43.         For i = 1 to (intNumberofIPs - 3)
  44.                 strPingIP = IncrementIP(strPingIP)
  45.                 Call PingHost(strPingIP)
  46.                 GetArp(strPingIP)
  47.                 Wscript.stdout.write vbcrlf
  48.         Next
  49. End Sub
  50.  
  51. sub PingHost(strPingIP)
  52.         Dim WshShell, WshPing, strPingResults
  53.         Wscript.StdOut.Write strPingIP
  54.         Set WshShell = WScript.CreateObject("WScript.Shell")
  55.         Set WshPing = WshShell.Exec("ping -n 3 -w 1000 " & strPingIP) 'send 3 echo requests, waiting 1 sec each
  56.         strPingResults = LCase(WshPing.StdOut.ReadAll)
  57.  
  58.         If InStr(strPingResults, "reply from") Then
  59.           WScript.StdOut.Write " is ALIVE"
  60. '               GetArp(strPingIP)
  61.         Else
  62.                 WScript.StdOut.Write " ping failed"
  63. '               getArp(strPingIP)
  64.         End If
  65. end Sub
  66.  
  67. Function GetArp(strPingIP)
  68.         Dim WshShell, WshArp, strArpResult
  69.         Set WshShell = WScript.CreateObject("WScript.Shell")
  70.         Set WshArp = WshShell.Exec("arp -a " & strPingIP)
  71.         strArpResult = WshArp.StdOut.ReadAll
  72.         Dim RegEx, Matches, Match
  73.         Set RegEx = New RegExp
  74.         RegEx.IgnoreCase = True
  75.         RegEx.Global = True
  76.         RegEx.Pattern = "[0-9A-F]{2}-[0-9A-F]{2}-[0-9A-F]{2}-[0-9A-F]{2}-[0-9A-F]{2}-[0-9A-F]{2}"
  77.         If Regex.Test(strArpResult) = True then
  78.           Wscript.StdOut.Write ", MAC(s)"
  79.                 Set Matches = RegEx.Execute(strArpResult)
  80.                 For each Match in Matches
  81.                         if not Match = "00-00-00-00-00-00" Then
  82.                                 wscript.StdOut.Write " " & Match
  83.                         Else wscript.stdout.write " -"
  84.                         End If
  85.                 Next
  86.         Else wscript.stdout.write ", no ARP."
  87.         End if
  88. End Function
  89.  
  90. Function CalcLastHostIP(strBroadcastIP)
  91.         Dim arrBroadcastIP
  92.         arrBroadcastIP = Split(strBroadcastIP,".")
  93.         CalcLastHostIP = arrBroadcastIP(0) &"."& arrBroadcastIP(1) &"."&arrBroadcastIP(2) &"."& (arrBroadcastIP(3) - 1)
  94. End Function
  95.  
  96. Function CalcBroadcastIP(strNetworkID,intNumberofIPs)
  97.         CalcBroadcastIP = strNetworkID
  98.         dim i
  99.         For i = 1 to (intNumberofIPs - 1)
  100.                 CalcBroadcastIP = IncrementIP(CalcBroadcastIP)
  101.         Next
  102. End Function
  103.  
  104. Function CalcNetIDandNumIPs(strIPaddress, strSubnetmask) 'check for proper subnet mask values; calc NetworkID; calc Number of IPs
  105. Dim arrReturn(2)
  106. Dim arrSubnetmask 'each octet is an element of the array
  107. Dim arrIPaddress'each octet is an element of the array
  108. arrSubnetmask = Split(strSubnetmask,".")
  109. arrIPaddress = Split(strIPaddress,".")
  110. Dim arrNetworkIP(3) 'the starting address for the subnet
  111. Dim arrBroadcastIP(3) 'the final (ending) address for the subnet
  112. Dim intOctet 'counter to determine which of the four octets (0 to 3) we are processing
  113. intOctet = 0
  114. Dim intNumIPsPerOctet 'number of IPs (or networks) allowed by the octet (a multiplier value)
  115. Dim intNumIPs 'number of hosts the mask allows (including broadcast IP & network address IP)
  116. intNumIPs = 1
  117.  
  118. Call SubnetmaskContainsProperValues(strSubnetmask, arrSubnetmask,0) 'test for proper values in subnet mask
  119. Do Until intOctet = 4   'parse each octet separately, check for proper values and calculate NetworkID & Number of IPs
  120.         Select Case arrSubnetmask(intOctet)
  121.         Case "0"               
  122.                 intNumIPsPerOctet = 256 - arrSubnetmask(intOctet)
  123.                 Call Subnetmask0FollowedBy0(strSubnetmask, arrSubnetmask, intOctet)
  124.                 arrNetworkIP(intOctet) = CalcNetworkID(arrIPAddress(intOctet), intNumIPsPerOctet)
  125.         Case "255"
  126.                 intNumIPsPerOctet = 256 - arrSubnetmask(intOctet)
  127.                 Call Subnetmask255PrecededBy255(strSubnetmask, arrSubnetmask, intOctet)
  128.                 arrNetworkIP(intOctet) = CalcNetworkID(arrIPAddress(intOctet), intNumIPsPerOctet)
  129.         Case Else
  130.                 intNumIPsPerOctet = 256 - arrSubnetmask(intOctet)
  131.                 Call SubnetmaskOthervalFollowedBy0s(strSubnetmask, arrSubnetmask, intOctet)
  132.                 Call SubnetmaskOthervalPrecededBy255s(strSubnetmask, arrSubnetmask, intOctet)
  133.                 arrNetworkIP(intOctet) = CalcNetworkID(arrIPAddress(intOctet), intNumIPsPerOctet)
  134.         End Select
  135.         intNumIPs = intNumIPs * IntNumIPsPerOctet
  136.         intOctet = IntOctet + 1
  137. Loop
  138. arrReturn(0) = arrNetworkIP(0) &"."& arrNetworkIP(1) &"."& arrNetworkIP(2) &"."& arrNetworkIP(3)
  139. arrReturn(1) = intNumIPs
  140. CalcNetIDandNumIPs = arrReturn
  141. end Function
  142.  
  143. Function CalcNetworkID(OctetVal, IntNumIPsPerOctet) 'calculate the network ID
  144.         Dim intWhichNet
  145.         Dim intStartAddress
  146.         intWhichNet = OctetVal \ intNumIpsperOctet 'integer result only; we don't care about the remainder
  147.         CalcNetworkID = intWhichNet * IntNumIPsPerOctet
  148. End Function
  149.  
  150. Function IncrementIP(strStartIP)
  151. 'takes IP as string, increments it by 1
  152.         Dim arrStartIP
  153.         arrStartIP = Split(strStartIP,".")
  154.                 if arrStartIP(3) < 255 then
  155.                         arrStartIP(3) = arrStartIP(3) + 1
  156.                 else
  157.                         if arrStartIP(2) < 255 then
  158.                                 arrStartIP(2) = arrStartIP(2) + 1
  159.                                 arrStartIP(3) = 0
  160.                         else
  161.                                 if arrStartIP(1) < 255 then
  162.                                         arrStartIP(1) = arrStartIP(1) + 1
  163.                                         arrStartIP(2) = 0
  164.                                         arrStartIP(3) = 0
  165.                                 else
  166.                                         if arrStartIP(0) < 255 then
  167.                                                 arrStartIP(0) = arrStartIP(0) + 1
  168.                                                 arrStartIP(2) = 0
  169.                                                 arrStartIP(2) = 0
  170.                                                 arrStartIP(3) = 0
  171.                                         else
  172.                                                 wscript.echo "Oops, cannot increment past 255.255.255.255 !"
  173.                                                 wscript.quit
  174.                                         end if
  175.                                 end if
  176.                         end if
  177.                 end if
  178.         IncrementIP = arrStartIP(0) &"."& arrStartIP(1) &"."& arrStartIP(2) &"."& arrStartIP(3)
  179. End Function
  180.  
  181. Sub SubnetmaskOthervalFollowedBy0s(strSubnetmask, arrSubnetmask, intOctet)
  182. 'all values following octet(iterator) that's not 255 or 0 must be 0
  183.         if intOctet = 3 Then Exit Sub 'don't perform check on last octet!
  184.         Dim iterator
  185.         iterator = intOctet + 1 
  186.         For iterator = intOctet + 1 to Ubound(arrSubnetmask)           
  187.                         if arrSubnetmask(iterator) <> 0 Then    
  188.                                 Wscript.Echo "Subnet Mask (" &strSubnetmask&  ") octet value {128, 192, 224, 240, 248, 252, 254}  is not followed by all zeros."
  189.                                 ExitBadSubnetMask(strSubnetMask)
  190.                         end if
  191.         Next
  192. end sub
  193.  
  194. Sub SubnetmaskOthervalPrecededBy255s(strSubnetmask, arrSubnetmask, intOctet)
  195.         'all values preceding octet(iterator) that's not 255 or 0 must be 255
  196.         if intOctet = 0 Then Exit Sub 'don't perform check on first octet!
  197.         Dim iterator
  198.         iterator = intOctet -1 
  199.         For iterator = Lbound(arrSubnetmask) to intOctet - 1
  200.                         if arrSubnetmask(iterator) <> 255 Then  
  201.                                 Wscript.Echo "Subnet Mask (" &strSubnetmask&  ") octet value {128, 192, 224, 240, 248, 252, 254}  is not preceded by all 255s."
  202.                                 ExitBadSubnetMask(strSubnetMask)
  203.                         end if
  204.         Next
  205. end Sub
  206.  
  207. Sub Subnetmask0FollowedBy0(strSubnetmask, arrSubnetmask, intOctet)
  208.         'octet value 0 must be followed by 0's
  209.         if intOctet = 3 Then Exit Sub 'don't perform check on last octet!
  210.         Dim iterator
  211.         iterator = intOctet + 1 
  212.         For iterator = (intOctet + 1) to Ubound(arrSubnetmask)         
  213.                         if arrSubnetmask(iterator) <> 0 Then    
  214.                                 Wscript.Echo "Subnet Mask (" &strSubnetmask&  ") octet value 0 is not followed by 0."
  215.                                 ExitBadSubnetMask(strSubnetMask)
  216.                         end if
  217.         Next
  218. end sub
  219.  
  220. Sub Subnetmask255PrecededBy255(strSubnetmask, arrSubnetmask, intOctet)
  221.         '255 must be preceeded by 255s
  222.         if intOctet = 0 Then Exit Sub 'don't perform check on first octet!
  223.         Dim iterator
  224.         For iterator = intOctet to Lbound(arrSubnetmask) step -1
  225.                         if arrSubnetmask(iterator) <> 255 Then  
  226.                                 Wscript.Echo "Subnet Mask (" &strSubnetmask&  ") octet value 255 is not preceded by all 255s."
  227.                                 ExitBadSubnetMask(strSubnetMask)
  228.                         end if
  229.         Next
  230. end sub
  231.  
  232. sub SubnetmaskContainsProperValues(strSubnetmask, arrSubnetmask, intOctet)
  233.         'There must be 4 octets
  234.         'and each octet must be one of: 0, 128, 192, 224, 240, 248, 252, 254, 255
  235.         Dim iterator
  236.         Dim arrSubnetProperValues
  237.         If Ubound(arrSubnetmask) <> 3 Then
  238.                 wscript.Echo "The subnet mask must contain 4 octets; given mask (" &strSubnetmask& ") does not."
  239.                 ExitBadSubnetMask(strSubnetmask)
  240.         End If
  241.         arrSubnetProperValues = Array(0, 128, 192, 224, 240, 248, 252, 254, 255)
  242.         Dim FoundAProperValue
  243.         FoundAProperValue = False
  244.         Dim SubnetProperValuesCounter
  245.         For iterator = Lbound(arrSubnetMask) to Ubound(arrSubnetmask)
  246.                 For SubnetProperValuesCounter = Lbound(arrSubnetProperValues) to Ubound(arrSubnetProperValues)
  247.                         if arrSubnetmask(iterator) = Cstr(arrSubnetProperValues(SubnetProperValuesCounter)) Then FoundAProperValue=True
  248.                 Next
  249.                 if not FoundAproperValue = True Then
  250.                         wscript.Echo "Each subnet mask octet must be one of: 0, 128, 192, 224, 240, 248, 252, 254, 255"
  251.                         ExitBadSubnetMask(strSubnetmask)
  252.                 End If
  253.                 FoundAProperValue=False
  254.         Next
  255.  
  256. end sub
  257.  
  258. sub ExitBadSubnetMask(strSubnetMask)
  259.         Wscript.Echo "Bad subnet mask " & strSubnetmask & " - exiting."
  260.         Wscript.Quit
  261. End Sub
  262.  
  263. sub HelpExit
  264.         Wscript.Echo "Usage example:"
  265.         Wscript.Echo
  266.         Wscript.Echo "cscript " & Wscript.ScriptName & " /IP:1.2.3.4 /mask:255.255.255.0 /ping"
  267.         wscript.Echo
  268.         Wscript.Echo "Where:"
  269.         Wscript.Echo "/IP: specifies a valid IP address and MUST be supplied."
  270.         Wscript.Echo "/mask: specifies a valid subnet mask and MUST be supplied"
  271.         Wscript.Echo "/ping is optional. If supplied, every host in the subnet will be pinged, and result shown."
  272.         Wscript.Echo
  273.         Wscript.Echo "Most invalid subnet masks will be flagged as such and the script will end."
  274.         Wscript.Echo "There is no checking for valid IP address yet!"
  275.         Wscript.Quit
  276. end sub

I purposely did not use any binary functions/math in building this script, because I wanted to get a better mental model of how subnetting works. And I'm just never going to be one of those people who does binary math in his head. So, after much fiddling, I arrived at the following method.

First I need to calculate the number of IP's that the subnet range will hold. To get started, I break the subnet mask down into its component octets (they are called octets because they're actually the decimal representation of an 8 bit binary number). If we have a subnet mask of 255.255.255.240, it looks like this:

255 255 255 240

I handle each octet individually. To figure out how many IPs/nets each octet will hold, the formula is simple: 256 - octet. So ..

256 - 255 = 1 256 - 255 = 1 256 - 255 = 1 256 - 240 = 16

Then I multiply them all together: 1 * 1 * 1 * 16 = 16

The subnet can hold 16 IP addresses. Great. Now I need to calculate the NetworkID, which is the first IP in the subnet. This isn't an IP you can realistically use for any host. To calculate the network ID, I use the formula

( IPoctet \ numberIPsPerOctet ) * numberIPsPerOctet

Take careful note of the backslash there. It signifies integer division, which is to say, we only want a whole number as a result. So, 5/2 = 2.5, but 5\2 = 2. We always round down to the nearest whole number. So even a result of 2.9999 would become ... 2. Here's how that plays out, for an IP of 10.0.0.42:

 
10 0 0 42
( 10 \ 1 ) * 1 = 10 ( 0 \ 1 ) * 1 = 0 ( 0 \ 1 ) * 1 = 0 ( 42 \ 16 ) * 16 = 32

 ... and so the resulting Network ID is ... drumroll please ... 10.0.0.32! So now we have:

  • Network ID = 10.0.0.32  (this is also the starting IP)
  • Total number of IPs in the subnet = 16

(Note that in the script, all of the above math is done in the function CalcNetIDandNumIPs, which calls CalcNetworkID at need. CalcNetIDandNumIPs also calls 4 different subroutines whose only job is to verify that the subnet mask is itself valid. But the code for those is fairly straightforward and they are well commented, so I will let you figure them out for yourself.

The hard work is mostly done. I can simply count forward from that starting address to get to the ending address in the range. Except ... what if the range is large enough that I need to increment through two or more octets? For instance, a subnet mask of 255.255.254.0 includes 512 addresses. I needed to write the IncrementIP function to take care of the special counting. It basically works like this:

If 4th octet is < 255, then add 1 to 4th octet, else
  If 3rd octet is <255, then add 1 to 3rd octet and set 4th octet to zero, else
    If 2nd octet is <255, then add 1 to 2nd octet and set 3rd and 4th octets to zero, else
       If 1st octet is <255, then add 1 to 1st octet and set 2nd, 3rd, and 4th octets to zero, else
         error! cannot increment past 255.255.255.255 ! (this should never happen anyway, because of earlier error checking)

So, to get the Broadcast IP (which is the last IP in the range), I just IncrementIP starting at the Network ID and going as many times as there are IPs in the subnet. CalcBroadcastIP handles this.

To get the first usable host IP, just add one to the 4th octet of the Network ID. To get the last usable host IP, subtract 1 from the last octet of the Broadcast IP.

All this seems simple now. But it took me 6-8 hours to figure it out and code it the first time! I'd appreciate any comments you might have, about how this could have been done better or more succinctly.

Simpler CMD version

Someone asked me for a simpler version in CMD. This one simply grabs the gateway address from ipconfig /all, assumes it is a class C address, and pings all IP's in the class C subnet of which that gateway is a member. It's not thoroughly tested but on quick run-thorough, it seems to do the job.

@echo off
setlocal
for /f "tokens=2 delims=:" %%a in ('ipconfig /all ^|findstr Gateway') do call :PINGSUBNET %%a
goto :eof
endlocal
:PINGSUBNET
set gateway=%1
for /f "tokens=1-3 delims=." %%a in ("%gateway%") do set prefix=%%a.%%b.%%c
if not "%prefix%"=="" for /l %%a in (1,1,254) do (set pingaddr=%prefix%.%%a) & call :PINGIP
goto :eof
:PINGIP
arp -d *
ping -n 1 -w 300 %pingaddr% >nul
for /f "tokens=2" %%a in ('arp -a ^| findstr %pingaddr%') do echo IP %pingaddr% .. ARP %%a
set pingaddr=

 


Guest  19 Jan 2010 
This is awesome.. but for some reason &apos does not work.. I had to replace it with ' BUT AWESOME nevertheless...
Guest  14 Sep 2014 
This is exactly what I was looking for. Thanks for putting this together.
sample@email.tst+(Guest)  08 Jun 2022 
1
sample@email.tst+(Guest)  23 Jul 2022 
1
sample@email.tst+(Guest)  26 Aug 2022 
1
1
sample@email.tst+(Guest)  21 Jan 2025 
1
Post a comment

Your Name or E-mail ID (mandatory)

 



 RSS of this page