2019-05-10 22:03:24 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
2018-02-18 17:42:17 +00:00
|
|
|
let
|
2021-07-06 08:12:28 +00:00
|
|
|
inherit (lib) mkOption types optionalString stringAfter;
|
2018-02-18 17:42:17 +00:00
|
|
|
|
2019-05-10 22:03:24 +00:00
|
|
|
cfg = config.boot.binfmt;
|
2018-02-18 17:42:17 +00:00
|
|
|
|
|
|
|
makeBinfmtLine = name: { recognitionType, offset, magicOrExtension
|
|
|
|
, mask, preserveArgvZero, openBinary
|
|
|
|
, matchCredentials, fixBinary, ...
|
|
|
|
}: let
|
|
|
|
type = if recognitionType == "magic" then "M" else "E";
|
|
|
|
offset' = toString offset;
|
|
|
|
mask' = toString mask;
|
|
|
|
interpreter = "/run/binfmt/${name}";
|
|
|
|
flags = if !(matchCredentials -> openBinary)
|
2019-05-10 22:03:24 +00:00
|
|
|
then throw "boot.binfmt.registrations.${name}: you can't specify openBinary = false when matchCredentials = true."
|
2018-02-18 17:42:17 +00:00
|
|
|
else optionalString preserveArgvZero "P" +
|
|
|
|
optionalString (openBinary && !matchCredentials) "O" +
|
|
|
|
optionalString matchCredentials "C" +
|
|
|
|
optionalString fixBinary "F";
|
|
|
|
in ":${name}:${type}:${offset'}:${magicOrExtension}:${mask'}:${interpreter}:${flags}";
|
|
|
|
|
2021-10-28 08:03:50 +00:00
|
|
|
activationSnippet = name: { interpreter, wrapInterpreterInShell, ... }: if wrapInterpreterInShell then ''
|
2020-12-31 19:28:01 +00:00
|
|
|
rm -f /run/binfmt/${name}
|
|
|
|
cat > /run/binfmt/${name} << 'EOF'
|
2021-01-25 04:47:59 +00:00
|
|
|
#!${pkgs.bash}/bin/sh
|
2020-12-31 19:28:01 +00:00
|
|
|
exec -- ${interpreter} "$@"
|
|
|
|
EOF
|
|
|
|
chmod +x /run/binfmt/${name}
|
2021-10-28 08:03:50 +00:00
|
|
|
'' else ''
|
|
|
|
rm -f /run/binfmt/${name}
|
|
|
|
ln -s ${interpreter} /run/binfmt/${name}
|
2020-12-31 19:28:01 +00:00
|
|
|
'';
|
2018-02-18 17:42:17 +00:00
|
|
|
|
2019-05-10 22:03:24 +00:00
|
|
|
getEmulator = system: (lib.systems.elaborate { inherit system; }).emulator pkgs;
|
2021-10-28 08:03:50 +00:00
|
|
|
getQemuArch = system: (lib.systems.elaborate { inherit system; }).qemuArch;
|
2018-02-18 17:42:17 +00:00
|
|
|
|
2019-05-11 01:00:21 +00:00
|
|
|
# Mapping of systems to “magicOrExtension” and “mask”. Mostly taken from:
|
|
|
|
# - https://github.com/cleverca22/nixos-configs/blob/master/qemu.nix
|
|
|
|
# and
|
|
|
|
# - https://github.com/qemu/qemu/blob/master/scripts/qemu-binfmt-conf.sh
|
|
|
|
# TODO: maybe put these in a JSON file?
|
|
|
|
magics = {
|
|
|
|
armv6l-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\x00\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
armv7l-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\x00\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
aarch64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xb7\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\x00\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
aarch64_be-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xb7'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
i386-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x03\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
i486-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x06\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
i586-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x06\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
i686-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x06\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
x86_64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x3e\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
alpha-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x26\x90'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xfe\xfe\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
sparc64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x02'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
sparc-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x12'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
powerpc-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x14'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
powerpc64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x15'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
powerpc64le-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x15\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\x00'';
|
|
|
|
};
|
|
|
|
mips-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
mipsel-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
mips64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'';
|
|
|
|
};
|
|
|
|
mips64el-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
riscv32-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xf3\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
riscv64-linux = {
|
|
|
|
magicOrExtension = ''\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xf3\x00'';
|
|
|
|
mask = ''\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'';
|
|
|
|
};
|
2019-07-17 21:00:46 +00:00
|
|
|
wasm32-wasi = {
|
|
|
|
magicOrExtension = ''\x00asm'';
|
|
|
|
mask = ''\xff\xff\xff\xff'';
|
|
|
|
};
|
|
|
|
wasm64-wasi = {
|
|
|
|
magicOrExtension = ''\x00asm'';
|
|
|
|
mask = ''\xff\xff\xff\xff'';
|
|
|
|
};
|
2019-05-11 01:00:21 +00:00
|
|
|
x86_64-windows = {
|
|
|
|
magicOrExtension = ".exe";
|
|
|
|
recognitionType = "extension";
|
|
|
|
};
|
|
|
|
i686-windows = {
|
|
|
|
magicOrExtension = ".exe";
|
|
|
|
recognitionType = "extension";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2019-05-10 22:03:24 +00:00
|
|
|
in {
|
2019-12-10 01:51:19 +00:00
|
|
|
imports = [
|
|
|
|
(lib.mkRenamedOptionModule [ "boot" "binfmtMiscRegistrations" ] [ "boot" "binfmt" "registrations" ])
|
|
|
|
];
|
|
|
|
|
2019-05-10 22:03:24 +00:00
|
|
|
options = {
|
|
|
|
boot.binfmt = {
|
|
|
|
registrations = mkOption {
|
|
|
|
default = {};
|
|
|
|
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
Extra binary formats to register with the kernel.
|
|
|
|
See https://www.kernel.org/doc/html/latest/admin-guide/binfmt-misc.html for more details.
|
|
|
|
'';
|
|
|
|
|
|
|
|
type = types.attrsOf (types.submodule ({ config, ... }: {
|
|
|
|
options = {
|
|
|
|
recognitionType = mkOption {
|
|
|
|
default = "magic";
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc "Whether to recognize executables by magic number or extension.";
|
2019-05-10 22:03:24 +00:00
|
|
|
type = types.enum [ "magic" "extension" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
offset = mkOption {
|
|
|
|
default = null;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc "The byte offset of the magic number used for recognition.";
|
2019-05-10 22:03:24 +00:00
|
|
|
type = types.nullOr types.int;
|
|
|
|
};
|
|
|
|
|
|
|
|
magicOrExtension = mkOption {
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc "The magic number or extension to match on.";
|
2019-05-10 22:03:24 +00:00
|
|
|
type = types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
mask = mkOption {
|
|
|
|
default = null;
|
|
|
|
description =
|
2022-07-28 21:19:15 +00:00
|
|
|
lib.mdDoc "A mask to be ANDed with the byte sequence of the file before matching";
|
2019-05-10 22:03:24 +00:00
|
|
|
type = types.nullOr types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
interpreter = mkOption {
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
The interpreter to invoke to run the program.
|
|
|
|
|
|
|
|
Note that the actual registration will point to
|
|
|
|
/run/binfmt/''${name}, so the kernel interpreter length
|
|
|
|
limit doesn't apply.
|
|
|
|
'';
|
|
|
|
type = types.path;
|
|
|
|
};
|
|
|
|
|
|
|
|
preserveArgvZero = mkOption {
|
|
|
|
default = false;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
Whether to pass the original argv[0] to the interpreter.
|
|
|
|
|
|
|
|
See the description of the 'P' flag in the kernel docs
|
|
|
|
for more details;
|
|
|
|
'';
|
|
|
|
type = types.bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
openBinary = mkOption {
|
|
|
|
default = config.matchCredentials;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
Whether to pass the binary to the interpreter as an open
|
|
|
|
file descriptor, instead of a path.
|
|
|
|
'';
|
|
|
|
type = types.bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
matchCredentials = mkOption {
|
|
|
|
default = false;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
Whether to launch with the credentials and security
|
|
|
|
token of the binary, not the interpreter (e.g. setuid
|
|
|
|
bit).
|
|
|
|
|
|
|
|
See the description of the 'C' flag in the kernel docs
|
|
|
|
for more details.
|
|
|
|
|
|
|
|
Implies/requires openBinary = true.
|
|
|
|
'';
|
|
|
|
type = types.bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
fixBinary = mkOption {
|
|
|
|
default = false;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
Whether to open the interpreter file as soon as the
|
|
|
|
registration is loaded, rather than waiting for a
|
|
|
|
relevant file to be invoked.
|
|
|
|
|
|
|
|
See the description of the 'F' flag in the kernel docs
|
|
|
|
for more details.
|
|
|
|
'';
|
|
|
|
type = types.bool;
|
|
|
|
};
|
2021-10-28 08:03:50 +00:00
|
|
|
|
|
|
|
wrapInterpreterInShell = mkOption {
|
|
|
|
default = true;
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2021-10-28 08:03:50 +00:00
|
|
|
Whether to wrap the interpreter in a shell script.
|
|
|
|
|
|
|
|
This allows a shell command to be set as the interpreter.
|
|
|
|
'';
|
|
|
|
type = types.bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
interpreterSandboxPath = mkOption {
|
|
|
|
internal = true;
|
|
|
|
default = null;
|
2022-08-28 23:38:36 +00:00
|
|
|
description = lib.mdDoc ''
|
2021-10-28 08:03:50 +00:00
|
|
|
Path of the interpreter to expose in the build sandbox.
|
|
|
|
'';
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
};
|
2018-02-18 17:42:17 +00:00
|
|
|
};
|
2019-05-10 22:03:24 +00:00
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
|
|
|
emulatedSystems = mkOption {
|
|
|
|
default = [];
|
2019-07-17 21:09:20 +00:00
|
|
|
example = [ "wasm32-wasi" "x86_64-windows" "aarch64-linux" ];
|
2022-07-28 21:19:15 +00:00
|
|
|
description = lib.mdDoc ''
|
2019-05-10 22:03:24 +00:00
|
|
|
List of systems to emulate. Will also configure Nix to
|
|
|
|
support your new systems.
|
2021-11-09 20:49:45 +00:00
|
|
|
Warning: the builder can execute all emulated systems within the same build, which introduces impurities in the case of cross compilation.
|
2019-05-10 22:03:24 +00:00
|
|
|
'';
|
2019-08-08 20:48:27 +00:00
|
|
|
type = types.listOf types.str;
|
2019-05-10 22:03:24 +00:00
|
|
|
};
|
2018-02-18 17:42:17 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2019-05-10 22:03:24 +00:00
|
|
|
config = {
|
|
|
|
boot.binfmt.registrations = builtins.listToAttrs (map (system: {
|
|
|
|
name = system;
|
2021-10-28 08:03:50 +00:00
|
|
|
value = let
|
2019-05-10 22:03:24 +00:00
|
|
|
interpreter = getEmulator system;
|
2021-10-28 08:03:50 +00:00
|
|
|
qemuArch = getQemuArch system;
|
|
|
|
|
|
|
|
preserveArgvZero = "qemu-${qemuArch}" == baseNameOf interpreter;
|
|
|
|
interpreterReg = let
|
|
|
|
wrapperName = "qemu-${qemuArch}-binfmt-P";
|
|
|
|
wrapper = pkgs.wrapQemuBinfmtP wrapperName interpreter;
|
|
|
|
in
|
|
|
|
if preserveArgvZero then "${wrapper}/bin/${wrapperName}"
|
|
|
|
else interpreter;
|
|
|
|
in {
|
|
|
|
inherit preserveArgvZero;
|
|
|
|
|
|
|
|
interpreter = interpreterReg;
|
|
|
|
wrapInterpreterInShell = !preserveArgvZero;
|
|
|
|
interpreterSandboxPath = dirOf (dirOf interpreterReg);
|
2019-05-11 01:00:21 +00:00
|
|
|
} // (magics.${system} or (throw "Cannot create binfmt registration for system ${system}"));
|
2019-05-10 22:03:24 +00:00
|
|
|
}) cfg.emulatedSystems);
|
2021-11-19 22:36:26 +00:00
|
|
|
nix.settings = lib.mkIf (cfg.emulatedSystems != []) {
|
|
|
|
extra-platforms = cfg.emulatedSystems ++ lib.optional pkgs.stdenv.hostPlatform.isx86_64 "i686-linux";
|
|
|
|
extra-sandbox-paths = let
|
2021-10-28 08:03:50 +00:00
|
|
|
ruleFor = system: cfg.registrations.${system};
|
|
|
|
hasWrappedRule = lib.any (system: (ruleFor system).wrapInterpreterInShell) cfg.emulatedSystems;
|
|
|
|
in [ "/run/binfmt" ]
|
|
|
|
++ lib.optional hasWrappedRule "${pkgs.bash}"
|
2021-11-19 22:36:26 +00:00
|
|
|
++ (map (system: (ruleFor system).interpreterSandboxPath) cfg.emulatedSystems);
|
|
|
|
};
|
2019-05-10 22:03:24 +00:00
|
|
|
|
|
|
|
environment.etc."binfmt.d/nixos.conf".source = builtins.toFile "binfmt_nixos.conf"
|
2019-05-11 02:53:14 +00:00
|
|
|
(lib.concatStringsSep "\n" (lib.mapAttrsToList makeBinfmtLine config.boot.binfmt.registrations));
|
2021-07-06 08:12:28 +00:00
|
|
|
system.activationScripts.binfmt = stringAfter [ "specialfs" ] ''
|
2019-05-10 22:03:24 +00:00
|
|
|
mkdir -p -m 0755 /run/binfmt
|
2019-05-11 02:53:14 +00:00
|
|
|
${lib.concatStringsSep "\n" (lib.mapAttrsToList activationSnippet config.boot.binfmt.registrations)}
|
2019-05-10 22:03:24 +00:00
|
|
|
'';
|
2020-05-13 14:29:34 +00:00
|
|
|
systemd.additionalUpstreamSystemUnits = lib.mkIf (config.boot.binfmt.registrations != {}) [
|
|
|
|
"proc-sys-fs-binfmt_misc.automount"
|
|
|
|
"proc-sys-fs-binfmt_misc.mount"
|
|
|
|
"systemd-binfmt.service"
|
|
|
|
];
|
2022-11-06 02:11:02 +00:00
|
|
|
systemd.services.systemd-binfmt.restartTriggers = [ (builtins.toJSON config.boot.binfmt.registrations) ];
|
2018-02-18 17:42:17 +00:00
|
|
|
};
|
|
|
|
}
|