2023-05-26 08:23:59 +00:00
|
|
|
import * as nl80211 from "nl80211";
|
|
|
|
import * as rtnl from "rtnl";
|
|
|
|
import { readfile } from "fs";
|
|
|
|
|
|
|
|
const iftypes = {
|
|
|
|
ap: nl80211.const.NL80211_IFTYPE_AP,
|
|
|
|
mesh: nl80211.const.NL80211_IFTYPE_MESH_POINT,
|
|
|
|
sta: nl80211.const.NL80211_IFTYPE_STATION,
|
|
|
|
adhoc: nl80211.const.NL80211_IFTYPE_ADHOC,
|
|
|
|
monitor: nl80211.const.NL80211_IFTYPE_MONITOR,
|
|
|
|
};
|
|
|
|
|
|
|
|
function wdev_remove(name)
|
|
|
|
{
|
|
|
|
nl80211.request(nl80211.const.NL80211_CMD_DEL_INTERFACE, 0, { dev: name });
|
|
|
|
}
|
|
|
|
|
|
|
|
function __phy_is_fullmac(phyidx)
|
|
|
|
{
|
|
|
|
let data = nl80211.request(nl80211.const.NL80211_CMD_GET_WIPHY, 0, { wiphy: phyidx });
|
|
|
|
|
|
|
|
return !data.software_iftypes.ap_vlan;
|
|
|
|
}
|
|
|
|
|
|
|
|
function phy_is_fullmac(phy)
|
|
|
|
{
|
|
|
|
let phyidx = int(trim(readfile(`/sys/class/ieee80211/${phy}/index`)));
|
|
|
|
|
|
|
|
return __phy_is_fullmac(phyidx);
|
|
|
|
}
|
|
|
|
|
|
|
|
function find_reusable_wdev(phyidx)
|
|
|
|
{
|
|
|
|
if (!__phy_is_fullmac(phyidx))
|
|
|
|
return null;
|
|
|
|
|
2023-08-02 11:18:03 +00:00
|
|
|
let data = nl80211.request(
|
2023-05-26 08:23:59 +00:00
|
|
|
nl80211.const.NL80211_CMD_GET_INTERFACE,
|
|
|
|
nl80211.const.NLM_F_DUMP,
|
|
|
|
{ wiphy: phyidx });
|
|
|
|
for (let res in data)
|
|
|
|
if (trim(readfile(`/sys/class/net/${res.ifname}/operstate`)) == "down")
|
|
|
|
return res.ifname;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function wdev_create(phy, name, data)
|
|
|
|
{
|
|
|
|
let phyidx = int(readfile(`/sys/class/ieee80211/${phy}/index`));
|
|
|
|
|
|
|
|
wdev_remove(name);
|
|
|
|
|
|
|
|
if (!iftypes[data.mode])
|
|
|
|
return `Invalid mode: ${data.mode}`;
|
|
|
|
|
|
|
|
let req = {
|
|
|
|
wiphy: phyidx,
|
|
|
|
ifname: name,
|
|
|
|
iftype: iftypes[data.mode],
|
|
|
|
};
|
|
|
|
|
|
|
|
if (data["4addr"])
|
|
|
|
req["4addr"] = data["4addr"];
|
|
|
|
if (data.macaddr)
|
|
|
|
req.mac = data.macaddr;
|
|
|
|
|
|
|
|
nl80211.error();
|
|
|
|
|
|
|
|
let reuse_ifname = find_reusable_wdev(phyidx);
|
|
|
|
if (reuse_ifname &&
|
|
|
|
(reuse_ifname == name ||
|
|
|
|
rtnl.request(rtnl.const.RTM_SETLINK, 0, { dev: reuse_ifname, ifname: name}) != false))
|
|
|
|
nl80211.request(
|
|
|
|
nl80211.const.NL80211_CMD_SET_INTERFACE, 0, {
|
|
|
|
wiphy: phyidx,
|
|
|
|
dev: name,
|
|
|
|
iftype: iftypes[data.mode],
|
|
|
|
});
|
|
|
|
else
|
|
|
|
nl80211.request(
|
|
|
|
nl80211.const.NL80211_CMD_NEW_INTERFACE,
|
|
|
|
nl80211.const.NLM_F_CREATE,
|
|
|
|
req);
|
|
|
|
|
|
|
|
let error = nl80211.error();
|
|
|
|
if (error)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
if (data.powersave != null) {
|
|
|
|
nl80211.request(nl80211.const.NL80211_CMD_SET_POWER_SAVE, 0,
|
|
|
|
{ dev: name, ps_state: data.powersave ? 1 : 0});
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-09-06 14:58:09 +00:00
|
|
|
function phy_sysfs_file(phy, name)
|
|
|
|
{
|
|
|
|
return trim(readfile(`/sys/class/ieee80211/${phy}/${name}`));
|
|
|
|
}
|
|
|
|
|
|
|
|
function macaddr_split(str)
|
|
|
|
{
|
|
|
|
return map(split(str, ":"), (val) => hex(val));
|
|
|
|
}
|
|
|
|
|
|
|
|
function macaddr_join(addr)
|
|
|
|
{
|
|
|
|
return join(":", map(addr, (val) => sprintf("%02x", val)));
|
|
|
|
}
|
|
|
|
|
|
|
|
function wdev_generate_macaddr(phy, data)
|
|
|
|
{
|
|
|
|
let idx = int(data.id ?? 0);
|
|
|
|
let mbssid = int(data.mbssid ?? 0) > 0;
|
|
|
|
let num_global = int(data.num_global ?? 1);
|
|
|
|
let use_global = !mbssid && idx < num_global;
|
|
|
|
|
|
|
|
let base_addr = phy_sysfs_file(phy, "macaddress");
|
|
|
|
if (!base_addr)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (!idx && !mbssid)
|
|
|
|
return base_addr;
|
|
|
|
|
|
|
|
let base_mask = phy_sysfs_file(phy, "address_mask");
|
|
|
|
if (!base_mask)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (base_mask == "00:00:00:00:00:00" && idx >= num_global) {
|
|
|
|
let addrs = split(phy_sysfs_file(phy, "addresses"), "\n");
|
|
|
|
|
|
|
|
if (idx < length(addrs))
|
|
|
|
return addrs[idx];
|
|
|
|
|
|
|
|
base_mask = "ff:ff:ff:ff:ff:ff";
|
|
|
|
}
|
|
|
|
|
|
|
|
let addr = macaddr_split(base_addr);
|
|
|
|
let mask = macaddr_split(base_mask);
|
|
|
|
let type;
|
|
|
|
|
|
|
|
if (mbssid)
|
|
|
|
type = "b5";
|
|
|
|
else if (use_global)
|
|
|
|
type = "add";
|
|
|
|
else if (mask[0] > 0)
|
|
|
|
type = "b1";
|
|
|
|
else if (mask[5] < 0xff)
|
|
|
|
type = "b5";
|
|
|
|
else
|
|
|
|
type = "add";
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case "b1":
|
|
|
|
if (!(addr[0] & 2))
|
|
|
|
idx--;
|
|
|
|
addr[0] |= 2;
|
|
|
|
addr[0] ^= idx << 2;
|
|
|
|
break;
|
|
|
|
case "b5":
|
|
|
|
if (mbssid)
|
|
|
|
addr[0] |= 2;
|
|
|
|
addr[5] ^= idx;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
for (let i = 5; i > 0; i--) {
|
|
|
|
addr[i] += idx;
|
|
|
|
if (addr[i] < 256)
|
|
|
|
break;
|
|
|
|
addr[i] %= 256;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return macaddr_join(addr);
|
|
|
|
}
|
|
|
|
|
2023-05-26 08:23:59 +00:00
|
|
|
const vlist_proto = {
|
|
|
|
update: function(values, arg) {
|
|
|
|
let data = this.data;
|
|
|
|
let cb = this.cb;
|
|
|
|
let seq = { };
|
|
|
|
let new_data = {};
|
|
|
|
let old_data = {};
|
|
|
|
|
|
|
|
this.data = new_data;
|
|
|
|
|
|
|
|
if (type(values) == "object") {
|
|
|
|
for (let key in values) {
|
|
|
|
old_data[key] = data[key];
|
|
|
|
new_data[key] = values[key];
|
|
|
|
delete data[key];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (let val in values) {
|
|
|
|
let cur_key = val[0];
|
|
|
|
let cur_obj = val[1];
|
|
|
|
|
|
|
|
old_data[cur_key] = data[cur_key];
|
|
|
|
new_data[cur_key] = val[1];
|
|
|
|
delete data[cur_key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let key in data) {
|
|
|
|
cb(null, data[key], arg);
|
|
|
|
delete data[key];
|
|
|
|
}
|
|
|
|
for (let key in new_data)
|
|
|
|
cb(new_data[key], old_data[key], arg);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
function is_equal(val1, val2) {
|
|
|
|
let t1 = type(val1);
|
|
|
|
|
|
|
|
if (t1 != type(val2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (t1 == "array") {
|
|
|
|
if (length(val1) != length(val2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (let i = 0; i < length(val1); i++)
|
|
|
|
if (!is_equal(val1[i], val2[i]))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else if (t1 == "object") {
|
|
|
|
for (let key in val1)
|
|
|
|
if (!is_equal(val1[key], val2[key]))
|
|
|
|
return false;
|
|
|
|
for (let key in val2)
|
2023-09-03 07:51:08 +00:00
|
|
|
if (val1[key] == null)
|
2023-05-26 08:23:59 +00:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return val1 == val2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function vlist_new(cb) {
|
|
|
|
return proto({
|
|
|
|
cb: cb,
|
|
|
|
data: {}
|
|
|
|
}, vlist_proto);
|
|
|
|
}
|
|
|
|
|
2023-09-06 14:58:09 +00:00
|
|
|
export { wdev_remove, wdev_create, wdev_generate_macaddr, is_equal, vlist_new, phy_is_fullmac };
|