IP.Mod 40 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757
  1. (* Aos, Copyright 2001, Pieter Muller, ETH Zurich *)
  2. MODULE IP; (** AUTHOR "pjm, mvt, eb"; PURPOSE "IP (v4 and v6)"; *)
  3. IMPORT SYSTEM, KernelLog, Commands, Strings, Network;
  4. CONST
  5. (* DEBUG *)
  6. DEBUG = TRUE;
  7. (** Error codes *)
  8. Ok* = 0;
  9. DeviceAlreadyUsed* = 3901;
  10. DuplicateInterfaceName* = 3902;
  11. NoInterfaceName* = 3903;
  12. GatewayNotInSubnet* = 3904;
  13. IPv6AdrUsedOnIPv4Interface* = 4001;
  14. IPv4AdrUsedOnIPv6Interface* = 4002;
  15. DublicatedAddresses* = 4003;
  16. MixedIpProtocols* = 4003;
  17. LocalAdrSetIntv6* = 4004; (* local address must be NIL when calling SetAdrs on a IPv6 interface *)
  18. PrefixNotSet* = 4005; (* prefix on ipv6 interface must be set with local address *)
  19. MaxNofDNS* = 10; (** maximum number of DNS servers per interface *)
  20. MaxTTL* = 255; (** maximum time-to-live on outgoing datagrams *)
  21. NbrOfReceivers = 255; (* Number of possible receivers, i.e. possible layer 3 protocols *)
  22. (** IP address constants *)
  23. NilAdrIPv4 = 0;
  24. (* Comparators for Adr.usedProtocols *)
  25. IPv4* = 4;
  26. IPv6* = 6;
  27. NilAdrIdent = -1; (* usedProtocol of NilAdrs *)
  28. (* ICMP *)
  29. ICMPHdrLen* = 4;
  30. ICMPSrcLLAdrOptionType* = 1;
  31. ICMPTargetLLAdrOptionType* = 2;
  32. ICMPPrefixInfoOptionType* = 3;
  33. ICMPRedirectHdrOptionType* = 4;
  34. ICMPMTUOptionType* = 5;
  35. ICMPAdvIntOptionType* = 7;
  36. ICMPHomeAgOptionType* = 8;
  37. ICMPRouteOption* = 9;
  38. TYPE
  39. Adr* = RECORD
  40. ipv4Adr*: LONGINT;
  41. ipv6Adr*: ARRAY 16 OF CHAR;
  42. usedProtocol*: LONGINT;
  43. data*: LONGINT;
  44. END; (** An IP Address. usedProtocol = 0: No protocol yet used
  45. usedProtocol = IPv4: IPv4 address stored in field ipv4Adr
  46. usedProtocol = IPv6: IPv6 address stored in field ipv6Adr
  47. data can be used to store additional informations. I.e. in IPv6 the
  48. prefix length is stored in the data field *)
  49. TYPE
  50. Packet* = POINTER TO ARRAY OF CHAR;
  51. Name* = ARRAY 128 OF CHAR; (** Name type for interface name *)
  52. Interface* = OBJECT
  53. VAR
  54. (** IP addresses of this interface. *)
  55. localAdr*, maskAdr*, gatewayAdr*, subnetAdr*, broadAdr*: Adr;
  56. (** name of the interface *)
  57. name*: Name;
  58. (** Device that the interface belongs to *)
  59. dev*: Network.LinkDevice;
  60. (** DNS server list - can be used by DNS, not used in IP itself *)
  61. DNS-: ARRAY MaxNofDNS OF Adr; (* DNS server list *)
  62. DNScount*: LONGINT; (* number of DNS servers in list *)
  63. (* interface *)
  64. next*: Interface; (* next pointer for interface list *)
  65. closed*: BOOLEAN; (* is interface already closed? *)
  66. protocol*: LONGINT; (* Interface for IPv4 or IPv6?. Only used by IP otherwise use dynamic type checking! *)
  67. (** Set addresses. Is normally called just after instanciation, but can also be called later, e.g. by DHCP.
  68. If "gatewayAdr" is "NilAdr", the subnet is considered to have no gateway, else it must be in the same
  69. subnet as the "localAdr".
  70. "domain" can be an empty string. It is normally used by a DNS implementation. It is not used in IP directly.
  71. In IPv6 maskAdr is the prefix of the currently IP address *)
  72. PROCEDURE SetAdrs*(localAdr, maskOrPrefixAdr, gatewayAdr: Adr; VAR res: WORD);
  73. BEGIN
  74. (* Extensions have to override this method.*)
  75. HALT(99);
  76. END SetAdrs;
  77. (** Remove all domain name servers from the interface. *)
  78. PROCEDURE DNSRemoveAll*;
  79. BEGIN {EXCLUSIVE}
  80. DNScount := 0;
  81. END DNSRemoveAll;
  82. (** Add a domain name server to the interface. *)
  83. PROCEDURE DNSAdd*(adr: Adr);
  84. VAR
  85. i: LONGINT;
  86. BEGIN {EXCLUSIVE}
  87. (* concurrent lookup is consistent *)
  88. ASSERT(DNScount < MaxNofDNS);
  89. i := 0;
  90. WHILE i < DNScount DO
  91. IF AdrsEqual (DNS[i],adr) THEN RETURN END;
  92. INC(i)
  93. END;
  94. DNS[DNScount] := adr;
  95. INC(DNScount);
  96. END DNSAdd;
  97. (** Remove a domain name server from the interface. *)
  98. PROCEDURE DNSRemove*(adr: Adr);
  99. VAR i: LONGINT;
  100. BEGIN {EXCLUSIVE}
  101. (* concurrent lookup could result in a duplicate address in worst-case *)
  102. (* this happends hardly ever and is harmless for DNS anyway *)
  103. i := 0;
  104. WHILE (i < DNScount) & (~ AdrsEqual(DNS[i],adr)) DO
  105. INC(i)
  106. END;
  107. IF i < DNScount THEN
  108. INC(i);
  109. WHILE i < DNScount DO
  110. DNS[i-1] := DNS[i];
  111. INC(i);
  112. END;
  113. DEC(DNScount);
  114. END;
  115. END DNSRemove;
  116. (** Send an IP packet on this interface. *)
  117. PROCEDURE Send*(type: LONGINT; destAdr: Adr; CONST l4hdr, data: ARRAY OF CHAR; h4len, dofs, dlen, TTL: LONGINT);
  118. BEGIN
  119. (* Extensions have to override this method.*)
  120. HALT(99);
  121. END Send;
  122. (* Internal procedure to perform the rest of the send operation. Used by "Send" and for IP forwarding. *)
  123. PROCEDURE DoSend*(destAdr: Adr; CONST l3hdr, l4hdr, data: ARRAY OF CHAR; h3len, h4len, dofs, dlen: LONGINT) ;
  124. BEGIN
  125. (* Extensions have to override this method.*)
  126. HALT(99);
  127. END DoSend;
  128. (** Enumerate all ARP table entries. *)
  129. PROCEDURE ARPEnumerate*(handle: ARPHandler);
  130. BEGIN
  131. (* Extensions have to override this method.*)
  132. HALT(99);
  133. END ARPEnumerate;
  134. (** Close and deactivate the interface, i.e. remove it from the configuration. *)
  135. PROCEDURE Close*;
  136. BEGIN
  137. (* Extensions have to override this method.*)
  138. HALT(99);
  139. END Close;
  140. (** Check if adr is a broadcast address *)
  141. PROCEDURE IsBroadcast*(adr: Adr) : BOOLEAN;
  142. BEGIN
  143. (* Extensions have to override this method.*)
  144. HALT(99);
  145. END IsBroadcast;
  146. (** Check if adr is a multicast address *)
  147. PROCEDURE IsMulticast*(adr: Adr) : BOOLEAN;
  148. BEGIN
  149. (* Extensions have to override this method.*)
  150. HALT(99);
  151. END IsMulticast;
  152. (** Receive an IP packet *)
  153. PROCEDURE IPInput*(dev: Network.LinkDevice; type: LONGINT; buffer: Network.Buffer);
  154. BEGIN
  155. (* Extensions have to override this method.*)
  156. HALT(99);
  157. END IPInput;
  158. (** Reads the source address of a IP packet buffer *)
  159. PROCEDURE ReadSrcAdr* (buffer: Network.Buffer): Adr;
  160. BEGIN
  161. (* Extensions have to override this method.*)
  162. HALT(99);
  163. END ReadSrcAdr;
  164. (** Reads the destination address of a IP packet buffer *)
  165. PROCEDURE ReadDestAdr* (buffer: Network.Buffer): Adr;
  166. BEGIN
  167. (* Extensions have to override this method.*)
  168. HALT(99);
  169. END ReadDestAdr;
  170. (** Creates a pseudo-header for checksum calculation (TCP/UDP) and returns the length of this header *)
  171. PROCEDURE WritePseudoHeader*(VAR pseudoHdr: ARRAY OF CHAR; src, dst: Adr; protocol, pktLengthUpperLayer: LONGINT): LONGINT;
  172. BEGIN
  173. (* Extensions have to override this method.*)
  174. HALT(99);
  175. END WritePseudoHeader;
  176. (** Writes the configuration of this interface *)
  177. PROCEDURE OutInterface*;
  178. BEGIN
  179. (* Extensions have to override this method.*)
  180. HALT(99);
  181. END OutInterface;
  182. END Interface;
  183. TYPE
  184. (* List of interfaces *)
  185. InterfaceList* = POINTER TO RECORD
  186. interface*: Interface;
  187. next*: InterfaceList;
  188. END;
  189. Receiver* = PROCEDURE {DELEGATE} (int: Interface; type: LONGINT; fip, lip: Adr; buffer: Network.Buffer);
  190. V6InterfaceByDstIP* = PROCEDURE {DELEGATE} (dstAdr: Adr): Interface;
  191. (** Handler for ARPPool.Enumerate. *)
  192. ARPHandler* = PROCEDURE {DELEGATE} (ip: Adr; complete: BOOLEAN; link: Network.LinkAdr; size, sendTime, updateTime, updateDate, hash: LONGINT);
  193. InterfaceHandler* = PROCEDURE {DELEGATE} (int: Interface);
  194. VAR
  195. (* receiver *)
  196. receivers*: ARRAY 256 OF Receiver; (* registered receivers - array position is IPv4 protocol field or IPv6 next header field *)
  197. (* Interface by dst for IPv6 *)
  198. v6InterfaceByDstIP*: V6InterfaceByDstIP;
  199. (* IP *)
  200. NilAdr*: Adr; (* To check if an IP address is NIL use IsNilAdr instead *)
  201. preferredProtocol*: LONGINT; (* Preferred IP protocol *)
  202. (* IP counters *)
  203. NIPSentToSubnet*, NIPSentToGateway*, NIPSentBroadcast*, NIPCantFragment*,
  204. NIPRcvTotal*, NIPTooSmall*, NIPBadVersion*, NIPOptions*, NIPBadChecksum*,
  205. NIPBadLength*, NIPTrim*, NIPBadHdrLen*, NIPNotForUs*, NIPCantReassemble*, NIPSrcIsBroadcast*,
  206. NIPDelivered*, NIPNoReceiver*, NIPForwarded*, NIPSentLocalLoopback*, NIPSentPointToPoint*: LONGINT;
  207. (* turn on/off IP forwarding, echo replay *)
  208. IPForwarding*: BOOLEAN;
  209. EchoReply*: BOOLEAN;
  210. (* Interface *)
  211. interfaces*: Interface; (* list of all installed interfaces *)
  212. counter: LONGINT; (* Use it only in the module body! *)
  213. (** Is address not yet specified *)
  214. PROCEDURE IsNilAdr* (adr: Adr): BOOLEAN;
  215. VAR
  216. isNil: BOOLEAN;
  217. i: LONGINT;
  218. BEGIN
  219. CASE adr.usedProtocol OF
  220. IPv4:
  221. RETURN (adr.ipv4Adr = NilAdrIPv4)
  222. |IPv6:
  223. isNil := TRUE;
  224. i := 0;
  225. WHILE ((i<16) & isNil) DO
  226. IF adr.ipv6Adr[i] # 0X THEN
  227. isNil := FALSE;
  228. END;
  229. INC(i);
  230. END;
  231. RETURN isNil;
  232. |NilAdrIdent:
  233. RETURN TRUE;
  234. ELSE
  235. IF DEBUG THEN
  236. ASSERT(TRUE);
  237. END;
  238. RETURN TRUE;
  239. END;
  240. END IsNilAdr;
  241. (* Checks if two addresses are equal *)
  242. PROCEDURE AdrsEqual* (adr1, adr2: Adr): BOOLEAN;
  243. VAR
  244. equal: BOOLEAN;
  245. i: LONGINT;
  246. BEGIN
  247. IF adr1.usedProtocol # adr2.usedProtocol THEN
  248. RETURN FALSE;
  249. END;
  250. CASE adr1.usedProtocol OF
  251. IPv4:
  252. IF adr1.ipv4Adr = adr2.ipv4Adr THEN
  253. RETURN TRUE;
  254. END;
  255. |IPv6:
  256. equal := TRUE;
  257. i := 0;
  258. WHILE ((i < 16) & equal) DO
  259. IF adr1.ipv6Adr[i] # adr2.ipv6Adr[i] THEN
  260. equal := FALSE;
  261. END;
  262. INC(i);
  263. END;
  264. IF adr1.data # adr2.data THEN
  265. equal := FALSE;
  266. END;
  267. RETURN equal;
  268. |NilAdrIdent:
  269. (* both addresses NIL therefore equal *)
  270. IF adr2.usedProtocol = NilAdrIdent THEN
  271. RETURN TRUE;
  272. ELSE
  273. RETURN FALSE;
  274. END;
  275. ELSE
  276. IF DEBUG THEN
  277. ASSERT(TRUE);
  278. END;
  279. RETURN FALSE;
  280. END;
  281. RETURN FALSE;
  282. END AdrsEqual;
  283. (** Convert a dotted-decimal string to an ip address. Return NilAdr on failure. *)
  284. PROCEDURE StrToAdr*(ipString: ARRAY OF CHAR): Adr;
  285. VAR
  286. retAdr: Adr;
  287. i, j, x: LONGINT;
  288. adr: ARRAY 4 OF CHAR;
  289. ok: BOOLEAN;
  290. charCount: LONGINT; (* ipv6: number of character between two : *)
  291. ipv6AdrPart: ARRAY 6 OF CHAR; (* two bytes of an IPv6 address *)
  292. ipv6AdrRight: ARRAY 16 OF CHAR; (* right part of an IPv6 address; after :: *)
  293. hexToChar: ARRAY 3 OF CHAR;
  294. leftParts: LONGINT; (* number of bytes before :: *)
  295. rightParts: LONGINT; (* number of bytes after :: *)
  296. val, res: LONGINT;
  297. state: LONGINT; (* state of the FSM look at the eof for more info *)
  298. dPointOcc: BOOLEAN; (* double point occured *)
  299. prefixVal: LONGINT;
  300. (* compute a subpart (two bytes) of a IPv6 address; subpart:=between two : *)
  301. PROCEDURE ComputeIPv6Part():BOOLEAN;
  302. BEGIN
  303. CASE charCount OF
  304. 0:
  305. RETURN TRUE;
  306. |1,2:
  307. IF dPointOcc THEN
  308. ipv6AdrRight[rightParts] := 0X;
  309. INC(rightParts);
  310. ELSE
  311. retAdr.ipv6Adr[leftParts] := 0X;
  312. INC(leftParts);
  313. END;
  314. Strings.HexStrToInt(ipv6AdrPart, val, res);
  315. IF res = Strings.Ok THEN
  316. IF dPointOcc THEN
  317. ipv6AdrRight[rightParts] := CHR(val);
  318. INC(rightParts);
  319. ELSE
  320. retAdr.ipv6Adr[leftParts] := CHR(val);
  321. INC(leftParts);
  322. END;
  323. ELSE
  324. RETURN FALSE;
  325. END;
  326. |3:
  327. hexToChar[0] := ipv6AdrPart[0];
  328. hexToChar[1] := 0X;
  329. Strings.HexStrToInt(hexToChar, val, res);
  330. IF res = Strings.Ok THEN
  331. IF dPointOcc THEN
  332. ipv6AdrRight[rightParts] := CHR(val);
  333. INC(rightParts);
  334. ELSE
  335. retAdr.ipv6Adr[leftParts] := CHR(val);
  336. INC(leftParts);
  337. END;
  338. ELSE
  339. RETURN FALSE;
  340. END;
  341. ipv6AdrPart[0] := "0";
  342. Strings.HexStrToInt(ipv6AdrPart, val, res);
  343. IF res = Strings.Ok THEN
  344. IF dPointOcc THEN
  345. ipv6AdrRight[rightParts] := CHR(val);
  346. INC(rightParts);
  347. ELSE
  348. retAdr.ipv6Adr[leftParts] := CHR(val);
  349. INC(leftParts);
  350. END;
  351. ELSE
  352. RETURN FALSE;
  353. END;
  354. |4:
  355. hexToChar[0] := ipv6AdrPart[0];
  356. hexToChar[1] := ipv6AdrPart[1];
  357. hexToChar[2] := 0X;
  358. Strings.HexStrToInt(hexToChar, val, res);
  359. IF res = Strings.Ok THEN
  360. IF dPointOcc THEN
  361. ipv6AdrRight[rightParts] := CHR(val);
  362. INC(rightParts);
  363. ELSE
  364. retAdr.ipv6Adr[leftParts] := CHR(val);
  365. INC(leftParts);
  366. END;
  367. ELSE
  368. RETURN FALSE;
  369. END;
  370. ipv6AdrPart[0] := "0";
  371. ipv6AdrPart[1] := "0";
  372. Strings.HexStrToInt(ipv6AdrPart, val, res);
  373. IF res = Strings.Ok THEN
  374. IF dPointOcc THEN
  375. ipv6AdrRight[rightParts] := CHR(val);
  376. INC(rightParts);
  377. ELSE
  378. retAdr.ipv6Adr[leftParts] := CHR(val);
  379. INC(leftParts);
  380. END;
  381. ELSE
  382. RETURN FALSE;
  383. END;
  384. ELSE
  385. IF DEBUG THEN
  386. ASSERT(TRUE);
  387. END;
  388. RETURN FALSE;
  389. END;
  390. charCount := 0;
  391. RETURN TRUE;
  392. END ComputeIPv6Part;
  393. BEGIN
  394. retAdr := NilAdr;
  395. IF IsValidIPv4Str(ipString) THEN
  396. (* Return an ipv4 address *)
  397. i := 0;
  398. j := 0;
  399. x := -1;
  400. ok := FALSE;
  401. LOOP
  402. IF (ipString[i] = ".") OR (ipString[i] = 0X) THEN
  403. IF (x < 0) OR (x > 255) OR (j = 4) THEN EXIT END;
  404. adr[j] := CHR(x);
  405. IF ipString[i] = 0X THEN ok := (j = 3); EXIT END;
  406. x := -1; INC(i); INC(j)
  407. ELSIF (ipString[i] >= "0") & (ipString[i] <= "9") THEN
  408. IF x = -1 THEN x := 0 END;
  409. x := x*10 + (ORD(ipString[i])-ORD("0"));
  410. INC(i)
  411. ELSE
  412. EXIT
  413. END
  414. END;
  415. IF ok THEN
  416. retAdr.ipv4Adr := SYSTEM.VAL (LONGINT, adr);
  417. retAdr.usedProtocol := IPv4;
  418. RETURN retAdr;
  419. ELSE
  420. RETURN NilAdr;
  421. END
  422. (* IPv6 *)
  423. ELSIF IsValidIPv6Str(ipString) THEN
  424. i := 0;
  425. state := 1;
  426. charCount := 0;
  427. dPointOcc := FALSE;
  428. retAdr.usedProtocol := 6;
  429. retAdr.ipv4Adr := NilAdrIPv4;
  430. i := 0;
  431. j := 0;
  432. charCount := 0;
  433. leftParts := 0;
  434. rightParts := 0;
  435. prefixVal := 0;
  436. Strings.UpperCase(ipString);
  437. WHILE (i < (LEN(ipString) - 1)) & (ipString[i] # 0X) DO
  438. CASE state OF (* Using the same FSM as IsValidIPv6Str *)
  439. -1: (* Error state
  440. Should never happen, is checked by IsValidIPv6Str() *)
  441. RETURN NilAdr;
  442. |1: (* reading two blocks of two bytes of 0-9\A-F *)
  443. IF ipString[i] = ":" THEN
  444. ipv6AdrPart[charCount] := 0X;
  445. IF ~ComputeIPv6Part() THEN
  446. RETURN NilAdr;
  447. END;
  448. state := 2;
  449. ELSIF ipString[i] = "/" THEN
  450. ipv6AdrPart[charCount] := 0X;
  451. IF ~ComputeIPv6Part() THEN
  452. RETURN NilAdr;
  453. END;
  454. state := 3;
  455. ELSE
  456. (* 0-9, A-F *)
  457. ipv6AdrPart[charCount] := ipString[i];
  458. INC(charCount);
  459. END;
  460. |2: (* a : occured *)
  461. IF ipString[i] = ":" THEN
  462. dPointOcc := TRUE;
  463. state := 4;
  464. ELSE
  465. (* 0-9, A-F *)
  466. state := 1;
  467. charCount := 0;
  468. ipv6AdrPart[charCount] := ipString[i];
  469. INC(charCount);
  470. END;
  471. |3: (* prefix will follow *)
  472. prefixVal := (prefixVal * 10) + (ORD(ipString[i]) - ORD("0"));
  473. |4: (* A :: occured *)
  474. IF ipString[i] = "/" THEN
  475. state := 3;
  476. ELSE
  477. IF ~ComputeIPv6Part() THEN
  478. RETURN NilAdr;
  479. END;
  480. (* 0-9, A-F *)
  481. state := 1;
  482. charCount := 0;
  483. ipv6AdrPart[charCount] := ipString[i];
  484. INC(charCount);
  485. END;
  486. ELSE
  487. IF DEBUG THEN
  488. ASSERT(TRUE);
  489. END;
  490. END;
  491. INC(i);
  492. END;
  493. ipv6AdrPart[charCount] := 0X;
  494. IF charCount # 0 THEN
  495. IF ~ComputeIPv6Part() THEN
  496. RETURN NilAdr;
  497. END;
  498. END;
  499. IF dPointOcc THEN
  500. (* fill 0X for :: *)
  501. FOR i:= leftParts TO ((LEN(retAdr.ipv6Adr) -1) - rightParts) DO
  502. retAdr.ipv6Adr[i] := 0X;
  503. END;
  504. (* fill part behind :: *)
  505. FOR i := 0 TO (rightParts - 1) DO
  506. retAdr.ipv6Adr[(LEN(retAdr.ipv6Adr) - rightParts) + i] := ipv6AdrRight[i];
  507. END;
  508. END;
  509. IF prefixVal > 64 THEN
  510. RETURN NilAdr;
  511. END;
  512. retAdr.data := prefixVal;
  513. RETURN retAdr;
  514. END;
  515. RETURN NilAdr;
  516. END StrToAdr;
  517. (** Convert an IP address to a dotted-decimal string. *)
  518. PROCEDURE AdrToStr*(adr: Adr; VAR string: ARRAY OF CHAR);
  519. VAR
  520. i, j, x: LONGINT;
  521. a: ARRAY 4 OF CHAR;
  522. val: LONGINT;
  523. hexToStr: ARRAY 5 OF CHAR;
  524. prefixLenStr: ARRAY 64 OF CHAR;
  525. maxZeroRow: LONGINT;
  526. currentZeroRow: LONGINT;
  527. maxZeroStart: LONGINT;
  528. currentZeroStart: LONGINT;
  529. lastZero: BOOLEAN;
  530. lastDPoint: BOOLEAN;
  531. countEnded: BOOLEAN;
  532. BEGIN
  533. CASE adr.usedProtocol OF
  534. IPv4:
  535. ASSERT(LEN(string) >= 16); (* enough space for largest result *)
  536. Network.Put4(a, 0, adr.ipv4Adr);
  537. i := 0;
  538. FOR j := 0 TO 3 DO
  539. x := ORD(a[j]);
  540. IF x >= 100 THEN string[i] := CHR(ORD("0")+x DIV 100); INC(i) END;
  541. IF x >= 10 THEN string[i] := CHR(ORD("0")+x DIV 10 MOD 10); INC(i) END;
  542. string[i] := CHR(ORD("0")+x MOD 10); INC(i);
  543. IF j = 3 THEN string[i] := 0X ELSE string[i] := "." END;
  544. INC(i)
  545. END
  546. |IPv6:
  547. FOR i := 0 TO (LEN(adr.ipv6Adr) -1) BY 2 DO
  548. (* simple version *)
  549. val := ORD(adr.ipv6Adr[i]) * 256;
  550. val := val + ORD(adr.ipv6Adr[i+1]);
  551. Strings.IntToHexStr (val, 3, hexToStr);
  552. (* Delete leading zeros *)
  553. WHILE (hexToStr[0] = "0") & (hexToStr[1] # 0X) DO
  554. Strings.Delete(hexToStr, 0, 1);
  555. END;
  556. Strings.Append (string, hexToStr);
  557. IF i # (LEN(adr.ipv6Adr) - 2) THEN
  558. Strings.Append (string, ":");
  559. END;
  560. END;
  561. (* replace longest row of zeros with :: *)
  562. maxZeroRow := 0;
  563. currentZeroRow := 0;
  564. maxZeroStart := 0;
  565. currentZeroStart := 0;
  566. i := 0;
  567. lastZero := FALSE;
  568. lastDPoint := TRUE;
  569. countEnded :=TRUE;
  570. WHILE string[i] # 0X DO
  571. IF string[i] = "0" THEN
  572. IF lastDPoint THEN
  573. INC(currentZeroRow);
  574. lastZero := TRUE;
  575. lastDPoint := FALSE;
  576. IF countEnded THEN
  577. currentZeroStart := i;
  578. countEnded := FALSE;
  579. END;
  580. END;
  581. ELSIF string[i] = ":" THEN
  582. lastDPoint := TRUE;
  583. IF lastZero THEN
  584. lastZero := FALSE;
  585. END;
  586. ELSE
  587. IF lastDPoint THEN
  588. lastDPoint := FALSE;
  589. countEnded := TRUE;
  590. IF currentZeroRow > maxZeroRow THEN
  591. maxZeroRow := currentZeroRow;
  592. maxZeroStart := currentZeroStart;
  593. END;
  594. END;
  595. END;
  596. INC(i);
  597. END;
  598. IF ~countEnded THEN
  599. IF currentZeroRow > maxZeroRow THEN
  600. maxZeroRow := currentZeroRow;
  601. maxZeroStart := currentZeroStart;
  602. END;
  603. END;
  604. IF maxZeroRow # 0 THEN
  605. (* write a :: *)
  606. IF maxZeroStart = 0 THEN
  607. string[0] := ":";
  608. i := 1;
  609. WHILE ((string[i] # 0X) & ~((string[i] # "0") & (string[i] # ":"))) DO INC(i); END;
  610. IF string[i] = 0X THEN
  611. string := "::";
  612. ELSE
  613. Strings.Delete(string, 1, i-2);
  614. END;
  615. ELSE
  616. i := maxZeroStart;
  617. WHILE ((string[i] = "0") OR (string[i] = ":")) DO INC(i); END;
  618. IF string[i] = 0X THEN
  619. string[maxZeroStart] := ":";
  620. string[maxZeroStart+1] := 0X;
  621. ELSE
  622. Strings.Delete(string, maxZeroStart, i - maxZeroStart - 1);
  623. END;
  624. END;
  625. END;
  626. IF adr.data # 0 THEN
  627. (* write prefix *)
  628. Strings.IntToStr(adr.data, prefixLenStr);
  629. Strings.Append (string, "/");
  630. Strings.Append (string, prefixLenStr);
  631. END;
  632. ELSE
  633. IF IsNilAdr (adr) THEN
  634. string := "";
  635. END;
  636. END;
  637. END AdrToStr;
  638. (** Convert a IP address from an array [ofs..ofs+x] to an
  639. Adr-type variable.
  640. Example for IPv4:
  641. If the LSB (least significant byte) is stored the the beginning [ofs],
  642. LSBfirst must be set to TRUE.
  643. (address "a.b.c.d" is stored as [d,c,b,a])
  644. If the LSB is stored at the end [ofs+3], LSBfirst must be set to FALSE.
  645. (address "a.b.c.d" is stored as [a,b,c,d])
  646. *)
  647. PROCEDURE ArrayToAdr*(CONST array: ARRAY OF CHAR; ofs, protocol: LONGINT; LSBfirst: BOOLEAN): Adr;
  648. VAR
  649. adr: Adr;
  650. i, swapTemp: LONGINT;
  651. BEGIN
  652. ASSERT((protocol = 4) OR (protocol = 6));
  653. IF protocol = IPv4 THEN (* index check *)
  654. IF ~(ofs + 4 <= LEN(array)) THEN
  655. RETURN NilAdr;
  656. END;
  657. SYSTEM.MOVE(ADDRESSOF(array[ofs]), ADDRESSOF(adr.ipv4Adr), 4);
  658. IF LSBfirst THEN
  659. SwapEndian(adr.ipv4Adr);
  660. END;
  661. adr.usedProtocol := IPv4;
  662. ELSIF protocol = IPv6 THEN
  663. IF ~(ofs + 16 <= LEN(array)) THEN
  664. RETURN NilAdr;
  665. END;
  666. SYSTEM.MOVE(ADDRESSOF(array[ofs]), ADDRESSOF(adr.ipv6Adr), 16);
  667. IF LSBfirst THEN
  668. FOR i := 0 TO 3 DO
  669. SYSTEM.MOVE(ADDRESSOF(adr.ipv6Adr[i*4]), ADDRESSOF(swapTemp), 4);
  670. SwapEndian(swapTemp);
  671. SYSTEM.MOVE(ADDRESSOF(swapTemp), ADDRESSOF(adr.ipv6Adr[i*4]), 4);
  672. END;
  673. END;
  674. adr.usedProtocol := IPv6;
  675. ELSE
  676. IF DEBUG THEN
  677. ASSERT(TRUE);
  678. END;
  679. RETURN NilAdr;
  680. END;
  681. RETURN adr;
  682. END ArrayToAdr;
  683. (** Convert an Adr-type variable into an array [ofs..ofs+x]
  684. Example in IPv4:
  685. If the LSB (least significant byte) should be stored the the
  686. beginning [ofs], LSBfirst must be set to TRUE.
  687. (address "a.b.c.d" is stored as [d,c,b,a])
  688. If the LSB should be stored at the end [ofs+3], LSBfirst must be set to FALSE.
  689. (address "a.b.c.d" is stored as [a,b,c,d])
  690. *)
  691. PROCEDURE AdrToArray*(adr: Adr; VAR array: ARRAY OF CHAR; ofs: LONGINT; LSBfirst: BOOLEAN);
  692. VAR
  693. tempAdr: Adr;
  694. i, swapTemp: LONGINT;
  695. BEGIN
  696. tempAdr := adr;
  697. CASE adr.usedProtocol OF
  698. IPv4:
  699. IF ~(ofs+4 <= LEN(array)) THEN
  700. tempAdr := NilAdr;
  701. END;
  702. IF LSBfirst THEN
  703. SwapEndian(tempAdr.ipv4Adr);
  704. END;
  705. SYSTEM.MOVE(ADDRESSOF(tempAdr.ipv4Adr), ADDRESSOF(array[ofs]), 4);
  706. | IPv6:
  707. IF ~(ofs + 16 <= LEN(array)) THEN
  708. tempAdr := NilAdr;
  709. END;
  710. IF LSBfirst THEN
  711. FOR i := 0 TO 3 DO
  712. SYSTEM.MOVE(ADDRESSOF(tempAdr.ipv6Adr[i*4]), ADDRESSOF(swapTemp), 4);
  713. SwapEndian(swapTemp);
  714. SYSTEM.MOVE(ADDRESSOF(swapTemp), ADDRESSOF(tempAdr.ipv6Adr[i*4]), 4);
  715. END;
  716. END;
  717. SYSTEM.MOVE(ADDRESSOF(adr.ipv6Adr), ADDRESSOF(array[ofs]), 16);
  718. ELSE
  719. IF DEBUG THEN
  720. ASSERT(TRUE);
  721. END;
  722. END;
  723. END AdrToArray;
  724. (** Reads the IP source address from a buffer *)
  725. PROCEDURE SrcAdrFromBuffer* (buffer: Network.Buffer): Adr;
  726. VAR
  727. i: LONGINT;
  728. adr: Adr;
  729. BEGIN
  730. CASE ORD(buffer.data[buffer.l3ofs]) DIV 16 OF
  731. IPv4:
  732. adr.usedProtocol := IPv4;
  733. SYSTEM.MOVE(ADDRESSOF(buffer.data[buffer.l3ofs+12]),ADDRESSOF(adr.ipv4Adr),4); (* data might be unaligned *)
  734. SetIPv6AdrNil(adr);
  735. RETURN adr;
  736. |IPv6:
  737. adr.usedProtocol := IPv6;
  738. FOR i := 0 TO 15 DO
  739. adr.ipv6Adr[i] := buffer.data[buffer.l3ofs+8 + i];
  740. END;
  741. adr.ipv4Adr := NilAdrIPv4;
  742. ELSE
  743. IF DEBUG THEN
  744. ASSERT(TRUE);
  745. END;
  746. RETURN NilAdr;
  747. END;
  748. RETURN NilAdr;
  749. END SrcAdrFromBuffer;
  750. (** Reads the IP destination address from a buffer *)
  751. PROCEDURE DestAdrFromBuffer* (buffer: Network.Buffer): Adr;
  752. VAR
  753. adr: Adr;
  754. i: LONGINT;
  755. BEGIN
  756. CASE ORD(buffer.data[buffer.l3ofs]) DIV 16 OF
  757. IPv4:
  758. adr.usedProtocol := IPv4;
  759. SYSTEM.MOVE(ADDRESSOF(buffer.data[buffer.l3ofs+16]),ADDRESSOF(adr.ipv4Adr),4); (* data might be unaligned *)
  760. SetIPv6AdrNil(adr);
  761. RETURN adr;
  762. |IPv6:
  763. adr.usedProtocol := IPv6;
  764. FOR i := 0 TO 15 DO
  765. adr.ipv6Adr[i] := buffer.data[buffer.l3ofs+24 + i];
  766. END;
  767. adr.ipv4Adr := NilAdrIPv4;
  768. ELSE
  769. IF DEBUG THEN
  770. ASSERT(TRUE);
  771. END;
  772. RETURN NilAdr;
  773. END;
  774. RETURN NilAdr;
  775. END DestAdrFromBuffer;
  776. (* Swap internal representation of an IP address from big to little endian or vice versa. *)
  777. PROCEDURE -SwapEndian(VAR adr: LONGINT);
  778. CODE
  779. #IF I386 THEN
  780. POP EAX
  781. MOV ECX, [EAX]
  782. XCHG CL, CH
  783. ROL ECX, 16
  784. XCHG CL, CH
  785. MOV [EAX], ECX
  786. #ELSIF AMD64 THEN
  787. POP RAX
  788. MOV ECX, [RAX]
  789. BSWAP ECX
  790. MOV [RAX], ECX
  791. #ELSIF ARM THEN
  792. LDMIA SP!, {R0}
  793. LDR R1, [R0, #0]
  794. REV R1, R1
  795. STR R1, [R0, #0]
  796. #ELSE
  797. unimplemented
  798. #END
  799. END SwapEndian;
  800. (** Write an IP address. *)
  801. PROCEDURE OutAdr*(adr: Adr);
  802. VAR
  803. s: ARRAY 64 OF CHAR;
  804. BEGIN
  805. AdrToStr(adr, s); KernelLog.String(s);
  806. END OutAdr;
  807. (** Enumerate all installed interfaces *)
  808. PROCEDURE Enumerate*(handler: InterfaceHandler);
  809. VAR item: Interface;
  810. BEGIN
  811. item := interfaces;
  812. WHILE item # NIL DO
  813. handler(item);
  814. item := item.next;
  815. END;
  816. END Enumerate;
  817. (** Output the configuration of the specified interface. Can be used as Handler in Enumerate(). *)
  818. PROCEDURE OutInterface*(interface: Interface);
  819. BEGIN
  820. interface.OutInterface;
  821. END OutInterface;
  822. (** Aos command: Output statistics and configuration of all installed interfaces. *)
  823. PROCEDURE IPConfig*(context : Commands.Context);
  824. BEGIN
  825. context.out.String("Interfaces:"); context.out.Ln;
  826. context.out.String("----------------------"); context.out.Ln;
  827. Enumerate(OutInterface);
  828. context.out.Ln;
  829. context.out.String("IP statistics:"); context.out.Ln;
  830. context.out.String("----------------------"); context.out.Ln;
  831. context.out.String("NIPSentToSubnet"); context.out.Int(NIPSentToSubnet,10);context.out.Ln;
  832. context.out.String("NIPSentToGateway"); context.out.Int(NIPSentToGateway,10);context.out.Ln;
  833. context.out.String("NIPSentBroadcast"); context.out.Int(NIPSentBroadcast,10);context.out.Ln;
  834. context.out.String("NIPCantFragment"); context.out.Int(NIPCantFragment,10);context.out.Ln;
  835. context.out.String("NIPRcvTotal"); context.out.Int(NIPRcvTotal,10);context.out.Ln;
  836. context.out.String("NIPTooSmall"); context.out.Int(NIPTooSmall,10);context.out.Ln;
  837. context.out.String("NIPBadVersion"); context.out.Int(NIPBadVersion,10);context.out.Ln;
  838. context.out.String("NIPOptions"); context.out.Int(NIPOptions,10);context.out.Ln;
  839. context.out.String("NIPBadChecksum"); context.out.Int(NIPBadChecksum,10);context.out.Ln;
  840. context.out.String("NIPBadLength"); context.out.Int(NIPBadLength,10);context.out.Ln;
  841. context.out.String("NIPTrim"); context.out.Int(NIPTrim, 10);context.out.Ln;
  842. context.out.String("NIPBadHdrLen"); context.out.Int(NIPBadHdrLen,10);context.out.Ln;
  843. context.out.String("NIPNotForUs"); context.out.Int(NIPNotForUs,10);context.out.Ln;
  844. context.out.String("NIPCantReassemble"); context.out.Int(NIPCantReassemble,10);context.out.Ln;
  845. context.out.String("NIPSrcIsBroadcast"); context.out.Int(NIPSrcIsBroadcast,10);context.out.Ln;
  846. context.out.String("NIPDelivered"); context.out.Int(NIPDelivered,10);context.out.Ln;
  847. context.out.String("NIPNoReceiver"); context.out.Int(NIPNoReceiver,10);context.out.Ln;
  848. context.out.String("NIPForwarded"); context.out.Int(NIPForwarded,10);context.out.Ln;
  849. context.out.String("NIPSentLocalLoopback"); context.out.Int(NIPSentLocalLoopback,10);context.out.Ln;
  850. context.out.String("NIPSentPointToPoint"); context.out.Int(NIPSentPointToPoint,10);context.out.Ln; context.out.Ln;
  851. END IPConfig;
  852. (** First part of a split checksum operation. len must be non-zero and even. chk1 must be 0 at the
  853. beginning. The result can be passed to a following call to Checksum1(). The last chk1 result must
  854. be passed to Checksum2, which sums the data following sequentially on the data summed by this
  855. operation. Carries from bit 15 are accumulated in the top 16 bits, so at most 65535 words can be
  856. summed by Checksum1 and Checksum2 together. *)
  857. PROCEDURE Checksum1*(CONST data: ARRAY OF CHAR; ofs, len, chk1: LONGINT): LONGINT;
  858. CODE
  859. #IF I386 THEN
  860. MOV EAX, [EBP+chk1]
  861. MOV EBX, [EBP+data]
  862. ADD EBX, [EBP+ofs]
  863. MOV ECX, [EBP+len]
  864. SHR ECX, 1
  865. JNC sloop
  866. PUSH 8 ; ASSERT(~ODD(len))
  867. INT 3
  868. sloop:
  869. MOVZX EDX, WORD [EBX]
  870. ADD EAX, EDX
  871. ADD EBX, 2
  872. DEC ECX
  873. JNZ sloop
  874. #ELSIF AMD64 THEN
  875. MOV EAX, [RBP+chk1]
  876. MOV RBX, [RBP+data]
  877. ADD EBX, [RBP+ofs]
  878. MOV ECX, [RBP+len]
  879. SHR ECX, 1
  880. JNC sloop
  881. PUSH 8 ; ASSERT(~ODD(len))
  882. INT 3
  883. sloop:
  884. MOVZX EDX, WORD [RBX]
  885. ADD EAX, EDX
  886. ADD RBX, 2
  887. DEC ECX
  888. JNZ sloop
  889. #ELSIF ARM THEN
  890. LDR R0, [FP, #chk1]
  891. LDR R1, [FP, #ofs]
  892. LDR R2, [FP, #len]
  893. LDR R3, [FP, #data] ; LDR R3, [FP, #20]
  894. ADD R3, R3, R1
  895. AND R4, R2, #1
  896. LSR R2, R2, #1 ; number of 16-bit words to process
  897. CMP R4, #0
  898. BEQ WordLoop
  899. SWI #8 ; ASSERT(~ODD(length))
  900. WordLoop: ; sum up the words
  901. LDRH R4, [R3, #0]
  902. ADD R0, R0, R4
  903. ADD R3, R3, #2
  904. SUB R2, R2, #1
  905. CMP R2, #0
  906. BNE WordLoop
  907. #ELSE
  908. unimplemented
  909. #END
  910. END Checksum1;
  911. (** Continue a checksum operation. len can be zero or odd. chk1 can be the result of Checksum1 for
  912. the immediately preceding data, or 0 for contiguous data. *)
  913. PROCEDURE Checksum2*(CONST data: ARRAY OF CHAR; ofs, len, chk1: LONGINT): INTEGER;
  914. CODE
  915. #IF I386 THEN
  916. MOV EAX, [EBP+chk1]
  917. MOV EBX, [EBP+data]
  918. ADD EBX, [EBP+ofs]
  919. MOV ECX, [EBP+len]
  920. SHR ECX, 1
  921. JZ eloop
  922. sloop:
  923. MOVZX EDX, WORD [EBX]
  924. ADD EAX, EDX
  925. ADD EBX, 2
  926. DEC ECX
  927. JNZ sloop
  928. eloop:
  929. TEST [EBP+len], 1 ; check for odd length
  930. JZ even
  931. MOVZX EDX, BYTE [EBX] ; add last byte
  932. ADD EAX, EDX
  933. even:
  934. MOV ECX, EAX ; now add in all the carries
  935. SHR ECX, 16
  936. AND EAX, 0FFFFH
  937. ADD EAX, ECX
  938. MOV ECX, EAX ; add in additional carry
  939. SHR ECX, 16
  940. ADD EAX, ECX ; result is in low 16 bits of EAX
  941. NOT EAX
  942. #ELSIF AMD64 THEN
  943. MOV EAX, [RBP + chk1]
  944. MOV RBX, [RBP + data]
  945. ADD EBX, [RBP + ofs]
  946. MOV ECX, [RBP + len]
  947. SHR ECX, 1
  948. JZ eloop
  949. sloop:
  950. MOVZX EDX, WORD [RBX]
  951. ADD EAX, EDX
  952. ADD RBX, 2
  953. DEC ECX
  954. JNZ sloop
  955. eloop:
  956. TEST [RBP + len], 1 ; check for odd length
  957. JZ even
  958. MOVZX EDX, BYTE [RBX] ; add last byte
  959. ADD EAX, EDX
  960. even:
  961. MOV ECX, EAX ; now add in all the carries
  962. SHR ECX, 16
  963. AND EAX, 0FFFFH
  964. ADD EAX, ECX
  965. MOV ECX, EAX ; add in additional carry
  966. SHR ECX, 16
  967. ADD EAX, ECX ; result is in low 16 bits of EAX
  968. NOT EAX
  969. #ELSIF ARM THEN
  970. LDR R0, [FP, #chk1]
  971. LDR R1, [FP, #20] ; LDR R1, [FP, #data]
  972. LDR R2, [FP, #ofs]
  973. LDR R3, [FP, #len]
  974. ADD R1, R1, R2
  975. LSR R4, R3, #1 ; number of 16-bit words to process
  976. CMP R4, #0
  977. BEQ Remainder
  978. WordLoop: ; sum up the words
  979. LDRH R5, [R1, #0]
  980. ADD R0, R0, R5
  981. ADD R1, R1, #2
  982. SUB R4, R4, #1
  983. CMP R4, #0
  984. BNE WordLoop
  985. ANDS R5, R3, #1 ; if len is even
  986. BEQ WrapUp
  987. Remainder: ; addition of the remaining byte
  988. LDRB R5, [R1, #0]
  989. ADD R0, R0, R5
  990. WrapUp: ; 32-bit sum is in R0
  991. LDR R7, [PC, #WordMask - $ - 8] ; load the 16-bit word mask
  992. LSR R6, R0, 16 ; upper 16 bits of R0 is the carry
  993. AND R0, R0, R7
  994. ADD R0, R0, R6
  995. LSR R6, R0, 16 ; upper 16 bits of R0 is the carry
  996. AND R0, R0, R7
  997. ADD R0, R0, R6
  998. EOR R0, R0, R7 ; negate the bits of the resulted 16-bit word
  999. B end
  1000. data:
  1001. WordMask: d32 0FFFFH
  1002. end:
  1003. #ELSE
  1004. unimplemented
  1005. #END
  1006. END Checksum2;
  1007. (* Return TRUE if "adr1" and "adr2" are in the same subnet defined by "mask". Only for IPv4 *)
  1008. PROCEDURE SameSubnetv4(adr1, adr2, mask: LONGINT): BOOLEAN;
  1009. CODE
  1010. #IF I386 THEN
  1011. MOV EAX, [EBP + adr1]
  1012. MOV EBX, [EBP + adr2]
  1013. MOV ECX, [EBP + mask]
  1014. AND EAX, ECX
  1015. AND EBX, ECX
  1016. CMP EAX, EBX
  1017. SETE AL
  1018. #ELSIF AMD64 THEN
  1019. MOV EAX, [RBP + adr1]
  1020. MOV EBX, [RBP + adr2]
  1021. MOV ECX, [RBP + mask]
  1022. AND EAX, ECX
  1023. AND EBX, ECX
  1024. CMP EAX, EBX
  1025. SETE AL
  1026. #ELSIF ARM THEN
  1027. LDR R0, [FP, #adr1]
  1028. LDR R1, [FP, #adr2]
  1029. LDR R2, [FP, #mask]
  1030. AND R0, R0, R2
  1031. AND R1, R1, R2
  1032. CMP R0, R1
  1033. MOVEQ R0, #1
  1034. MOVNE R0, #0
  1035. #ELSE
  1036. unimplemented
  1037. #END
  1038. END SameSubnetv4;
  1039. (* Return TRUE if adr matches the prefix *)
  1040. PROCEDURE MatchPrefix*(adr: Adr; prefix: Adr): BOOLEAN;
  1041. VAR
  1042. bytesToCheck: LONGINT;
  1043. bitsToCheck: LONGINT;
  1044. i: LONGINT;
  1045. matches: BOOLEAN;
  1046. diffSet: SET;
  1047. BEGIN
  1048. IF DEBUG THEN
  1049. ASSERT ((IsNilAdr(adr)) OR (adr.usedProtocol = IPv6));
  1050. END;
  1051. matches := TRUE;
  1052. bytesToCheck := prefix.data DIV 8;
  1053. bitsToCheck := prefix.data MOD 8;
  1054. FOR i := 0 TO bytesToCheck - 1 DO
  1055. IF adr.ipv6Adr[i] # prefix.ipv6Adr[i] THEN
  1056. matches := FALSE;
  1057. END;
  1058. END;
  1059. IF bitsToCheck # 0 THEN
  1060. diffSet := {};
  1061. FOR i := 0 TO 8 - bitsToCheck - 1 DO
  1062. diffSet := diffSet + {i};
  1063. END;
  1064. FOR i := 0 TO bitsToCheck - 1 DO
  1065. IF (SYSTEM.VAL(SET, adr.ipv6Adr[bytesToCheck]) - diffSet) # (SYSTEM.VAL(SET, prefix.ipv6Adr[bytesToCheck]) - diffSet) THEN
  1066. matches := FALSE;
  1067. END;
  1068. END;
  1069. END;
  1070. RETURN matches;
  1071. END MatchPrefix;
  1072. (** Return the interface on which packets with "dst" address should be sent. Return NIL if no interface matches. *)
  1073. PROCEDURE InterfaceByDstIP*(dest: Adr): Interface;
  1074. VAR
  1075. item, gw: Interface;
  1076. BEGIN
  1077. CASE dest.usedProtocol OF
  1078. IPv4:
  1079. gw := NIL;
  1080. item := interfaces;
  1081. LOOP
  1082. IF item = NIL THEN EXIT END;
  1083. IF (item.protocol = IPv4) & (~IsNilAdr(item.localAdr)) & (item.dev.Linked() # Network.LinkNotLinked) THEN
  1084. IF SameSubnetv4(dest.ipv4Adr, item.subnetAdr.ipv4Adr, item.maskAdr.ipv4Adr) THEN
  1085. EXIT;
  1086. ELSIF (gw = NIL) & (~IsNilAdr(item.subnetAdr)) THEN
  1087. IF item.protocol # IPv4 THEN
  1088. gw := item;
  1089. ELSIF ~IsNilAdr(item.gatewayAdr) THEN
  1090. gw := item;
  1091. END;
  1092. END;
  1093. END;
  1094. item := item.next;
  1095. END;
  1096. IF item # NIL THEN
  1097. RETURN item;
  1098. ELSE
  1099. RETURN gw;
  1100. END;
  1101. |IPv6:
  1102. (* Requires special handling when multiple devices are present *)
  1103. RETURN v6InterfaceByDstIP(dest);
  1104. ELSE
  1105. IF DEBUG THEN
  1106. ASSERT(TRUE);
  1107. END;
  1108. RETURN NIL;
  1109. END;
  1110. END InterfaceByDstIP;
  1111. (** Return the reference to an installed interface by its name. NIL is returned if no interface with this name
  1112. was found. *)
  1113. PROCEDURE InterfaceByName*(CONST name: ARRAY OF CHAR): Interface;
  1114. VAR item: Interface;
  1115. BEGIN
  1116. item := interfaces;
  1117. WHILE (item # NIL) & (item.name # name) DO
  1118. item := item.next;
  1119. END;
  1120. RETURN item;
  1121. END InterfaceByName;
  1122. (** Delivers first preferred (if possible) interface of an device.
  1123. Return the reference to an installed interface by its device. NIL is returned if no interface with this
  1124. device was found. *)
  1125. PROCEDURE InterfaceByDevice*(dev: Network.LinkDevice): Interface;
  1126. VAR
  1127. item: Interface;
  1128. unprefInt: Interface;
  1129. BEGIN
  1130. unprefInt := NIL;
  1131. item := interfaces;
  1132. WHILE (item # NIL) DO
  1133. IF item.dev = dev THEN
  1134. IF item.protocol # preferredProtocol THEN
  1135. unprefInt := item;
  1136. ELSE
  1137. RETURN item;
  1138. END;
  1139. END;
  1140. item := item.next;
  1141. END;
  1142. (* return interface with unpreferred protocol or NIL *)
  1143. RETURN unprefInt;
  1144. END InterfaceByDevice;
  1145. (** Return a list of installed interfaces of a certain device. *)
  1146. PROCEDURE InterfaceListByDevice* (dev: Network.LinkDevice):InterfaceList;
  1147. VAR
  1148. item: Interface;
  1149. interfaceList: InterfaceList;
  1150. interfaceListItem: InterfaceList;
  1151. BEGIN
  1152. item := interfaces;
  1153. interfaceList := NIL;
  1154. (* search for interfaces with matching device and put them in a list *)
  1155. WHILE item # NIL DO
  1156. IF item.dev = dev THEN
  1157. NEW(interfaceListItem);
  1158. interfaceListItem.interface := item;
  1159. interfaceListItem.next := interfaceList;
  1160. interfaceList := interfaceListItem;
  1161. END;
  1162. item := item.next;
  1163. END;
  1164. RETURN interfaceList;
  1165. END InterfaceListByDevice;
  1166. (** Install a receiver for a type. The type is stored in
  1167. IPv4: Protocol field
  1168. IPv6: Next header field *)
  1169. PROCEDURE InstallReceiver*( type: LONGINT; r: Receiver);
  1170. BEGIN {EXCLUSIVE}
  1171. IF DEBUG THEN
  1172. ASSERT(r # NIL);
  1173. ASSERT((type >=0) & (type <= 255));
  1174. ASSERT(receivers[type] = NIL);
  1175. END;
  1176. receivers[type] := r;
  1177. END InstallReceiver;
  1178. (** Remove the currently installed receiver for a type. The type is stored in
  1179. IPv4: Protocol field
  1180. IPv6: Next header field *)
  1181. PROCEDURE RemoveReceiver*(type: LONGINT);
  1182. BEGIN {EXCLUSIVE}
  1183. ASSERT((type >=0) & (type <= 255));
  1184. ASSERT(receivers[type] # NIL);
  1185. receivers[type] := NIL;
  1186. END RemoveReceiver;
  1187. (** Checks if a string is a valid IPv4 address *)
  1188. PROCEDURE IsValidIPv4Str (CONST ipString: ARRAY OF CHAR): BOOLEAN;
  1189. VAR
  1190. i,j: LONGINT;
  1191. ipNr: LONGINT;
  1192. digits: ARRAY 4 OF CHAR;
  1193. startClass: LONGINT;
  1194. BEGIN
  1195. i := 0;
  1196. (* Class A *)
  1197. WHILE (i < Strings.Length(ipString)) & (ipString[i] #'.') & (i < 3) DO
  1198. digits[i] := ipString[i];
  1199. INC (i);
  1200. END;
  1201. digits[i] := 0X;
  1202. IF ipString[i] # '.' THEN RETURN FALSE END;
  1203. (* Check if in digits are only numbers *)
  1204. j := 0;
  1205. WHILE digits[j] # 0X DO
  1206. IF (ORD(digits[j]) - ORD("0")) > 9 THEN RETURN FALSE END;
  1207. INC (j);
  1208. END;
  1209. Strings.StrToInt (digits, ipNr);
  1210. IF ipNr > 255 THEN RETURN FALSE END;
  1211. (* Class B *)
  1212. INC(i);
  1213. startClass := i;
  1214. WHILE (i < Strings.Length(ipString)) & (ipString[i] # '.') & (i - startClass <= 3) DO
  1215. digits[i-startClass] := ipString[i];
  1216. INC (i);
  1217. END;
  1218. digits[i-startClass] := 0X;
  1219. IF ipString[i] # '.' THEN RETURN FALSE END;
  1220. (* Check if in digits are only number *)
  1221. j := 0;
  1222. WHILE digits[j] # 0X DO
  1223. IF (ORD(digits[j]) - ORD("0")) > 9 THEN RETURN FALSE END;
  1224. INC (j);
  1225. END;
  1226. Strings.StrToInt (digits, ipNr);
  1227. IF ipNr > 255 THEN RETURN FALSE END;
  1228. (* Class C *)
  1229. INC(i);
  1230. startClass := i;
  1231. WHILE (i < Strings.Length (ipString)) & (ipString[i] # '.') & (i - startClass <= 3) DO
  1232. digits[i-startClass] := ipString[i];
  1233. INC (i);
  1234. END;
  1235. digits[i-startClass] := 0X;
  1236. IF ipString[i] # '.' THEN RETURN FALSE END;
  1237. (* Check if in digits are only number *)
  1238. j := 0;
  1239. WHILE digits[j] # 0X DO
  1240. IF (ORD(digits[j]) - ORD("0")) > 9 THEN RETURN FALSE END;
  1241. INC (j);
  1242. END;
  1243. Strings.StrToInt (digits, ipNr);
  1244. IF ipNr > 255 THEN RETURN FALSE END;
  1245. (* Class D *)
  1246. INC(i);
  1247. startClass := i;
  1248. WHILE (i < Strings.Length (ipString)) & (i - startClass <= 3) DO
  1249. digits[i-startClass] := ipString[i];
  1250. INC (i);
  1251. END;
  1252. digits[i-startClass] := 0X;
  1253. (* Check if in digits are only number *)
  1254. j := 0;
  1255. WHILE digits[j] # 0X DO
  1256. IF (ORD(digits[j]) - ORD("0")) > 9 THEN RETURN FALSE END;
  1257. INC (j);
  1258. END;
  1259. Strings.StrToInt (digits, ipNr);
  1260. IF ipNr > 255 THEN RETURN FALSE END;
  1261. RETURN TRUE;
  1262. END IsValidIPv4Str;
  1263. (** Checks if a string is a valid IPv6 address *)
  1264. PROCEDURE IsValidIPv6Str (ipString: ARRAY OF CHAR): BOOLEAN;
  1265. VAR
  1266. i: LONGINT;
  1267. state: LONGINT; (* -1: error *)
  1268. charCount: LONGINT;
  1269. ascD: LONGINT;
  1270. ascH: LONGINT;
  1271. dPointOcc: BOOLEAN;
  1272. prefixLenArr: ARRAY 3 OF LONGINT;
  1273. prefixLen: LONGINT;
  1274. BEGIN
  1275. i := 0;
  1276. state := 1;
  1277. dPointOcc := FALSE;
  1278. Strings.UpperCase(ipString);
  1279. WHILE (i < (LEN(ipString) - 1)) & (ipString[i] # 0X) DO
  1280. CASE state OF
  1281. -1:
  1282. RETURN FALSE;
  1283. |1:
  1284. (* 0-9 & A-F *)
  1285. ascD := ORD(ipString[i]) - ORD("0");
  1286. ascH := ORD(ipString[i]) - ORD("A");
  1287. IF ((ascD >= 0) & (ascD <= 9)) OR ((ascH >= 0) & (ascH <= 5)) THEN
  1288. INC(charCount);
  1289. (* more than 4 digits between two : *)
  1290. IF charCount > 4 THEN
  1291. state := -1;
  1292. END;
  1293. (* : *)
  1294. ELSIF ipString[i] = ":" THEN
  1295. charCount := 0;
  1296. state := 2;
  1297. ELSIF ipString[i] = "/" THEN
  1298. charCount := 0;
  1299. state := 3;
  1300. ELSE
  1301. state := -1;
  1302. END;
  1303. |2:
  1304. ascD := ORD(ipString[i]) - ORD("0");
  1305. ascH := ORD(ipString[i]) - ORD("A");
  1306. IF ipString[i] = ":" THEN
  1307. IF dPointOcc THEN
  1308. state := -1;
  1309. ELSE
  1310. dPointOcc := TRUE;
  1311. state := 4;
  1312. END
  1313. ELSIF ((ascD >= 0) & (ascD <= 9)) OR ((ascH >= 0) & (ascH <= 5)) THEN
  1314. INC(charCount);
  1315. state := 1;
  1316. ELSE
  1317. state := -1;
  1318. END;
  1319. |3:
  1320. ascD := ORD(ipString[i]) - ORD("0");
  1321. IF ~((ascD >= 0) & (ascD <= 9)) THEN
  1322. state := -1;
  1323. ELSE
  1324. IF charCount > 3 THEN
  1325. state := -1;
  1326. ELSE
  1327. prefixLenArr[charCount] := ascD;
  1328. INC(charCount);
  1329. END;
  1330. END;
  1331. |4:
  1332. ascD := ORD(ipString[i]) - ORD("0");
  1333. ascH := ORD(ipString[i]) - ORD("A");
  1334. IF ipString[i] = "/" THEN
  1335. state := 3;
  1336. ELSIF ((ascD >= 0) & (ascD <= 9)) OR ((ascH >= 0) & (ascH <= 5)) THEN
  1337. INC(charCount);
  1338. state := 1;
  1339. ELSE
  1340. state := -1;
  1341. END;
  1342. ELSE
  1343. IF DEBUG THEN
  1344. ASSERT(TRUE);
  1345. END;
  1346. END;
  1347. INC(i);
  1348. END;
  1349. CASE state OF
  1350. 1:
  1351. RETURN TRUE;
  1352. |3:
  1353. IF charCount > 0 THEN
  1354. prefixLen := 0;
  1355. FOR i:= 0 TO charCount - 1 DO
  1356. prefixLen := prefixLen * 10;
  1357. INC(prefixLen, prefixLenArr[i]);
  1358. END;
  1359. IF prefixLen <= 64 THEN
  1360. RETURN TRUE;
  1361. ELSE
  1362. RETURN FALSE;
  1363. END;
  1364. ELSE
  1365. RETURN FALSE;
  1366. END;
  1367. |4:
  1368. RETURN TRUE;
  1369. ELSE
  1370. IF DEBUG THEN
  1371. ASSERT(TRUE);
  1372. END;
  1373. RETURN FALSE;
  1374. END;
  1375. RETURN FALSE;
  1376. END IsValidIPv6Str;
  1377. (** Set IPv6 address to zero *)
  1378. PROCEDURE SetIPv6AdrNil (adr: Adr);
  1379. VAR
  1380. i: LONGINT;
  1381. BEGIN
  1382. FOR i := 0 TO 15 DO
  1383. adr.ipv6Adr[i] := 0X;
  1384. END;
  1385. END SetIPv6AdrNil;
  1386. (* Add the interface to the IP configuration. *)
  1387. PROCEDURE AddInterface*(int: Interface; VAR res: WORD);
  1388. VAR
  1389. item: Interface;
  1390. BEGIN {EXCLUSIVE}
  1391. item := interfaces;
  1392. WHILE item # NIL DO
  1393. (* Check if interface name already exists *)
  1394. IF item.name = int.name THEN
  1395. res := DuplicateInterfaceName;
  1396. RETURN;
  1397. END;
  1398. item := item.next;
  1399. END;
  1400. (* Add interface *)
  1401. int.next := interfaces;
  1402. interfaces := int;
  1403. res := Ok;
  1404. END AddInterface;
  1405. (* Remove the interface from the IP configuration. *)
  1406. PROCEDURE RemoveInterface*(int: Interface);
  1407. VAR
  1408. item: Interface;
  1409. BEGIN {EXCLUSIVE}
  1410. item := interfaces;
  1411. IF item = NIL THEN
  1412. (* empty list *)
  1413. ELSIF item = int THEN
  1414. (* remove first item *)
  1415. interfaces := interfaces.next;
  1416. ELSE
  1417. WHILE item.next # int DO
  1418. item := item.next;
  1419. END;
  1420. IF item.next # NIL THEN
  1421. item.next := item.next.next;
  1422. ELSE
  1423. (* not found *)
  1424. END;
  1425. END;
  1426. END RemoveInterface;
  1427. (* Prints out a packet *)
  1428. PROCEDURE PacketOut*(CONST title, buffer: ARRAY OF CHAR; all: BOOLEAN);
  1429. VAR
  1430. i: LONGINT;
  1431. length: LONGINT;
  1432. BEGIN
  1433. KernelLog.Ln;KernelLog.String("********************");KernelLog.Ln;
  1434. KernelLog.String(title); KernelLog.Ln;
  1435. IF all THEN
  1436. length := LEN(buffer) -1;
  1437. ELSE
  1438. length := Strings.Min(LEN(buffer) - 1, 256);
  1439. END;
  1440. FOR i := 0 TO length DO
  1441. IF (i MOD 4) = 0 THEN
  1442. KernelLog.Ln;
  1443. KernelLog.Int(i, 2);KernelLog.String(": ");
  1444. END;
  1445. KernelLog.Hex(ORD(buffer[i]), -1);
  1446. KernelLog.String(" ");
  1447. END;
  1448. KernelLog.Ln;KernelLog.String("********************");KernelLog.Ln;
  1449. END PacketOut;
  1450. BEGIN
  1451. (* IPFowarding default value is FALSE *)
  1452. IPForwarding := FALSE;
  1453. (* EchoReply default value is TRUE *)
  1454. EchoReply := TRUE;
  1455. (* Initializations *)
  1456. (* NilAdr *)
  1457. NilAdr.ipv4Adr := NilAdrIPv4;
  1458. SetIPv6AdrNil (NilAdr);
  1459. NilAdr.usedProtocol := NilAdrIdent;
  1460. (* Initialize receiver list *)
  1461. FOR counter:=0 TO NbrOfReceivers-1 DO
  1462. receivers[counter] := NIL;
  1463. END;
  1464. (* Initialize inteface list *)
  1465. interfaces := NIL;
  1466. END IP.
  1467. Free:
  1468. System.Free TraceRoute VNC Ping WMFTPClient FTPClient WebFTPServer TCPServices TLS InitNetwork Ping DHCP TCP DNS UDP ICMP IPv4 IPv6 IP~
  1469. Start:
  1470. InitNetwork.Init
  1471. Compile:
  1472. PC.Compile \s IP.Mod IPv4.Mod IPv6.Mod ICMP.Mod UDP.Mod DNS.Mod TCP.Mod DHCP.Mod InitNetwork.Mod WebFTPServer.Mod FTPClient.Mod WMFTPClient.Mod Ping.Mod VNC.Mod TraceRoute.Mod~
  1473. History:
  1474. 02.05.2005 eb Supports IPv6 and fragmented IPv6 packets.
  1475. IsValidIPv6Str:
  1476. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~`
  1477. FSM used in IsValidIPv6Str:
  1478. -----------------------
  1479. EOS: end of string
  1480. State 1: Initial state
  1481. 0-9/A-F goto state 1
  1482. : goto state 2
  1483. / goto state 3
  1484. EOS valid
  1485. State 2: 0-9/A-F goto state 1
  1486. : goto state 4
  1487. EOS invalid
  1488. State 3: 0-9 goto state 3
  1489. EOS valid
  1490. State 4: / goto state 3
  1491. 0-9/A-F goto state 1
  1492. EOS valid