shithub: mc

ref: 334532827595c538c638dcae7ebecb4d857573f5
dir: /libstd/ipparse.myr/

View raw version
use "types.use"
use "intparse.use"
use "option.use"

 /* FIXME: needed for decls which should be pulled in as hidden */
use "strcmp.use"
use "utf.use"
use "fmt.use"

pkg std =
	const ip4parse	: (ip : byte[:] -> option(byte[4]))
	const ip6parse	: (ip : byte[:] -> option(byte[16]))
;;

const ip4parse = {ip
	var addr
	var last : size
	var x : option(int32)
	var val : int32 /* need int32 to check for overflow */
	var i
	var j : size

	i = 0
	for j = 0; j < ip.len; j++
		if ip[j] == '.' castto(byte)
			put("seg[%z..%z] = %s\n", last, j, ip[last:j])
			match intparsebase(ip[last:j], 10)
			| `Some v:
				val = v
				if val < 0 || val > 255
					-> `None
				;;
				addr[i++] = val castto(byte)
				last = j + 1
			| `None:
				-> `None
			;;
		;;
	;;
	match intparsebase(ip[last:j], 10)
	| `Some v:
		val = v
		if val < 0 || val > 255
			-> `None
		;;
		addr[i] = val castto(byte)
	| `None:
		-> `None
	;;
	if j != ip.len
		-> `None
	;;
	-> `Some addr
}