2020-11-04 11:53:50 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import copy
|
|
|
|
import lark
|
|
|
|
import os
|
|
|
|
import re
|
2020-11-23 15:14:29 +01:00
|
|
|
import xml.etree.ElementTree
|
2020-11-04 11:53:50 +01:00
|
|
|
|
|
|
|
|
2020-11-23 15:14:24 +01:00
|
|
|
def translate_vendor(name):
|
|
|
|
T = {
|
|
|
|
"CPUID_VENDOR_AMD": "AMD",
|
|
|
|
"CPUID_VENDOR_INTEL": "Intel",
|
|
|
|
"CPUID_VENDOR_HYGON": "Hygon",
|
|
|
|
}
|
|
|
|
|
|
|
|
if name in T:
|
|
|
|
return T[name]
|
|
|
|
|
|
|
|
print("warning: Unknown vendor '{}'".format(name))
|
|
|
|
return name
|
|
|
|
|
|
|
|
|
2020-11-23 15:14:25 +01:00
|
|
|
def translate_feature(name):
|
|
|
|
T = {
|
|
|
|
"CPUID_6_EAX_ARAT": "arat",
|
|
|
|
"CPUID_7_0_EBX_ADX": "adx",
|
|
|
|
"CPUID_7_0_EBX_AVX2": "avx2",
|
|
|
|
"CPUID_7_0_EBX_AVX512BW": "avx512bw",
|
|
|
|
"CPUID_7_0_EBX_AVX512CD": "avx512cd",
|
|
|
|
"CPUID_7_0_EBX_AVX512DQ": "avx512dq",
|
|
|
|
"CPUID_7_0_EBX_AVX512ER": "avx512er",
|
|
|
|
"CPUID_7_0_EBX_AVX512F": "avx512f",
|
2020-11-23 15:14:28 +01:00
|
|
|
"CPUID_7_0_EBX_AVX512IFMA": "avx512ifma",
|
2020-11-23 15:14:25 +01:00
|
|
|
"CPUID_7_0_EBX_AVX512PF": "avx512pf",
|
|
|
|
"CPUID_7_0_EBX_AVX512VL": "avx512vl",
|
|
|
|
"CPUID_7_0_EBX_BMI1": "bmi1",
|
|
|
|
"CPUID_7_0_EBX_BMI2": "bmi2",
|
|
|
|
"CPUID_7_0_EBX_CLFLUSHOPT": "clflushopt",
|
|
|
|
"CPUID_7_0_EBX_CLWB": "clwb",
|
|
|
|
"CPUID_7_0_EBX_ERMS": "erms",
|
|
|
|
"CPUID_7_0_EBX_FSGSBASE": "fsgsbase",
|
|
|
|
"CPUID_7_0_EBX_HLE": "hle",
|
|
|
|
"CPUID_7_0_EBX_INVPCID": "invpcid",
|
|
|
|
"CPUID_7_0_EBX_MPX": "mpx",
|
|
|
|
"CPUID_7_0_EBX_RDSEED": "rdseed",
|
|
|
|
"CPUID_7_0_EBX_RTM": "rtm",
|
|
|
|
"CPUID_7_0_EBX_SHA_NI": "sha-ni",
|
|
|
|
"CPUID_7_0_EBX_SMAP": "smap",
|
|
|
|
"CPUID_7_0_EBX_SMEP": "smep",
|
|
|
|
"CPUID_7_0_ECX_AVX512BITALG": "avx512bitalg",
|
|
|
|
"CPUID_7_0_ECX_AVX512_VBMI2": "avx512vbmi2",
|
|
|
|
"CPUID_7_0_ECX_AVX512_VBMI": "avx512vbmi",
|
|
|
|
"CPUID_7_0_ECX_AVX512VNNI": "avx512vnni",
|
|
|
|
"CPUID_7_0_ECX_AVX512_VPOPCNTDQ": "avx512-vpopcntdq",
|
|
|
|
"CPUID_7_0_ECX_CLDEMOTE": "cldemote",
|
|
|
|
"CPUID_7_0_ECX_GFNI": "gfni",
|
|
|
|
"CPUID_7_0_ECX_LA57": "la57",
|
|
|
|
"CPUID_7_0_ECX_MOVDIR64B": "movdir64b",
|
|
|
|
"CPUID_7_0_ECX_MOVDIRI": "movdiri",
|
|
|
|
"CPUID_7_0_ECX_PKU": "pku",
|
|
|
|
"CPUID_7_0_ECX_RDPID": "rdpid",
|
|
|
|
"CPUID_7_0_ECX_UMIP": "umip",
|
|
|
|
"CPUID_7_0_ECX_VAES": "vaes",
|
|
|
|
"CPUID_7_0_ECX_VPCLMULQDQ": "vpclmulqdq",
|
|
|
|
"CPUID_7_0_EDX_ARCH_CAPABILITIES": "arch-capabilities",
|
|
|
|
"CPUID_7_0_EDX_AVX512_4FMAPS": "avx512-4fmaps",
|
|
|
|
"CPUID_7_0_EDX_AVX512_4VNNIW": "avx512-4vnniw",
|
|
|
|
"CPUID_7_0_EDX_CORE_CAPABILITY": "core-capability",
|
2020-11-23 15:14:28 +01:00
|
|
|
"CPUID_7_0_EDX_FSRM": "fsrm",
|
2020-11-23 15:14:25 +01:00
|
|
|
"CPUID_7_0_EDX_SPEC_CTRL": "spec-ctrl",
|
|
|
|
"CPUID_7_0_EDX_SPEC_CTRL_SSBD": "ssbd",
|
|
|
|
"CPUID_7_0_EDX_STIBP": "stibp",
|
|
|
|
"CPUID_7_1_EAX_AVX512_BF16": "avx512-bf16",
|
2021-06-10 09:45:41 +02:00
|
|
|
"CPUID_7_1_EAX_AVX_VNNI": "avx-vnni",
|
2021-02-22 13:20:06 +01:00
|
|
|
"CPUID_8000_0008_EBX_AMD_SSBD": "amd-ssbd",
|
2020-11-23 15:14:25 +01:00
|
|
|
"CPUID_8000_0008_EBX_CLZERO": "clzero",
|
|
|
|
"CPUID_8000_0008_EBX_IBPB": "ibpb",
|
2021-02-22 13:20:07 +01:00
|
|
|
"CPUID_8000_0008_EBX_IBRS": "ibrs",
|
2020-11-23 15:14:25 +01:00
|
|
|
"CPUID_8000_0008_EBX_STIBP": "amd-stibp",
|
|
|
|
"CPUID_8000_0008_EBX_WBNOINVD": "wbnoinvd",
|
|
|
|
"CPUID_8000_0008_EBX_XSAVEERPTR": "xsaveerptr",
|
|
|
|
"CPUID_ACPI": "acpi",
|
|
|
|
"CPUID_APIC": "apic",
|
|
|
|
"CPUID_CLFLUSH": "clflush",
|
|
|
|
"CPUID_CMOV": "cmov",
|
|
|
|
"CPUID_CX8": "cx8",
|
|
|
|
"CPUID_DE": "de",
|
|
|
|
"CPUID_EXT2_3DNOW": "3dnow",
|
|
|
|
"CPUID_EXT2_3DNOWEXT": "3dnowext",
|
|
|
|
"CPUID_EXT2_FFXSR": "fxsr_opt",
|
|
|
|
"CPUID_EXT2_LM": "lm",
|
|
|
|
"CPUID_EXT2_MMXEXT": "mmxext",
|
|
|
|
"CPUID_EXT2_NX": "nx",
|
|
|
|
"CPUID_EXT2_PDPE1GB": "pdpe1gb",
|
|
|
|
"CPUID_EXT2_RDTSCP": "rdtscp",
|
|
|
|
"CPUID_EXT2_SYSCALL": "syscall",
|
|
|
|
"CPUID_EXT3_3DNOWPREFETCH": "3dnowprefetch",
|
|
|
|
"CPUID_EXT3_ABM": "abm",
|
|
|
|
"CPUID_EXT3_CR8LEG": "cr8legacy",
|
|
|
|
"CPUID_EXT3_FMA4": "fma4",
|
|
|
|
"CPUID_EXT3_LAHF_LM": "lahf_lm",
|
|
|
|
"CPUID_EXT3_MISALIGNSSE": "misalignsse",
|
|
|
|
"CPUID_EXT3_OSVW": "osvw",
|
|
|
|
"CPUID_EXT3_PERFCORE": "perfctr_core",
|
|
|
|
"CPUID_EXT3_SSE4A": "sse4a",
|
|
|
|
"CPUID_EXT3_SVM": "svm",
|
|
|
|
"CPUID_EXT3_TBM": "tbm",
|
|
|
|
"CPUID_EXT3_XOP": "xop",
|
|
|
|
"CPUID_EXT_AES": "aes",
|
|
|
|
"CPUID_EXT_AVX": "avx",
|
|
|
|
"CPUID_EXT_CX16": "cx16",
|
|
|
|
"CPUID_EXT_F16C": "f16c",
|
|
|
|
"CPUID_EXT_FMA": "fma",
|
|
|
|
"CPUID_EXT_MOVBE": "movbe",
|
|
|
|
"CPUID_EXT_PCID": "pcid",
|
|
|
|
"CPUID_EXT_PCLMULQDQ": "pclmuldq",
|
|
|
|
"CPUID_EXT_POPCNT": "popcnt",
|
|
|
|
"CPUID_EXT_RDRAND": "rdrand",
|
|
|
|
"CPUID_EXT_SSE3": "pni",
|
|
|
|
"CPUID_EXT_SSE41": "sse4.1",
|
|
|
|
"CPUID_EXT_SSE42": "sse4.2",
|
|
|
|
"CPUID_EXT_SSSE3": "ssse3",
|
|
|
|
"CPUID_EXT_TSC_DEADLINE_TIMER": "tsc-deadline",
|
|
|
|
"CPUID_EXT_X2APIC": "x2apic",
|
|
|
|
"CPUID_EXT_XSAVE": "xsave",
|
|
|
|
"CPUID_FP87": "fpu",
|
|
|
|
"CPUID_FXSR": "fxsr",
|
|
|
|
"CPUID_MCA": "mca",
|
|
|
|
"CPUID_MCE": "mce",
|
|
|
|
"CPUID_MMX": "mmx",
|
|
|
|
"CPUID_MSR": "msr",
|
|
|
|
"CPUID_MTRR": "mtrr",
|
|
|
|
"CPUID_PAE": "pae",
|
|
|
|
"CPUID_PAT": "pat",
|
|
|
|
"CPUID_PGE": "pge",
|
|
|
|
"CPUID_PSE36": "pse36",
|
|
|
|
"CPUID_PSE": "pse",
|
|
|
|
"CPUID_SEP": "sep",
|
|
|
|
"CPUID_SSE2": "sse2",
|
|
|
|
"CPUID_SSE": "sse",
|
|
|
|
"CPUID_SS": "ss",
|
|
|
|
"CPUID_SVM_NPT": "npt",
|
|
|
|
"CPUID_SVM_NRIPSAVE": "nrip-save",
|
2021-03-04 08:43:09 +01:00
|
|
|
"CPUID_SVM_SVME_ADDR_CHK": "svme-addr-chk",
|
2020-11-23 15:14:25 +01:00
|
|
|
"CPUID_TSC": "tsc",
|
|
|
|
"CPUID_VME": "vme",
|
|
|
|
"CPUID_XSAVE_XGETBV1": "xgetbv1",
|
|
|
|
"CPUID_XSAVE_XSAVEC": "xsavec",
|
|
|
|
"CPUID_XSAVE_XSAVEOPT": "xsaveopt",
|
|
|
|
"CPUID_XSAVE_XSAVES": "xsaves",
|
|
|
|
"MSR_ARCH_CAP_IBRS_ALL": "ibrs-all",
|
|
|
|
"MSR_ARCH_CAP_MDS_NO": "mds-no",
|
|
|
|
"MSR_ARCH_CAP_PSCHANGE_MC_NO": "pschange-mc-no",
|
|
|
|
"MSR_ARCH_CAP_RDCL_NO": "rdctl-no",
|
|
|
|
"MSR_ARCH_CAP_SKIP_L1DFL_VMENTRY": "skip-l1dfl-vmentry",
|
|
|
|
"MSR_ARCH_CAP_TAA_NO": "taa-no",
|
|
|
|
"MSR_CORE_CAP_SPLIT_LOCK_DETECT": "split-lock-detect",
|
|
|
|
}
|
|
|
|
|
2020-11-23 15:14:27 +01:00
|
|
|
ignore = any([
|
|
|
|
name.startswith("VMX_"),
|
|
|
|
name.startswith("vmx-"),
|
|
|
|
name.startswith("MSR_VMX_"),
|
|
|
|
name in ("0", "model", "model-id", "stepping"),
|
|
|
|
name in ("CPUID_EXT_MONITOR", "monitor"),
|
|
|
|
name in ("MSR_VMX_BASIC_DUAL_MONITOR", "dual-monitor"),
|
|
|
|
name in ("CPUID_EXT3_TOPOEXT", "topoext"),
|
|
|
|
])
|
|
|
|
|
|
|
|
if ignore:
|
|
|
|
return None
|
|
|
|
|
2020-11-23 15:14:25 +01:00
|
|
|
if name in T:
|
|
|
|
return T[name]
|
|
|
|
|
2020-11-23 15:14:26 +01:00
|
|
|
for v in T.values():
|
|
|
|
if name.replace("-", "_") == v.replace("-", "_"):
|
|
|
|
return v
|
|
|
|
|
2020-11-23 15:14:25 +01:00
|
|
|
print("warning: Unknown feature '{}'".format(name))
|
|
|
|
return name
|
|
|
|
|
|
|
|
|
2020-11-04 11:53:50 +01:00
|
|
|
def readline_cont(f):
|
|
|
|
"""Read one logical line from a file `f` i.e. continues lines that end in
|
|
|
|
a backslash."""
|
|
|
|
|
|
|
|
line = f.readline()
|
|
|
|
while line.endswith("\\\n"):
|
|
|
|
line = line[:-2] + " " + f.readline()
|
|
|
|
return line
|
|
|
|
|
|
|
|
|
|
|
|
def read_builtin_x86_defs(filename):
|
|
|
|
"""Extract content between begin_mark and end_mark from file `filename` as
|
|
|
|
string, while expanding shorthand macros like "I486_FEATURES"."""
|
|
|
|
|
2021-06-07 14:01:24 +02:00
|
|
|
begin_mark = re.compile("^static( const)? X86CPUDefinition builtin_x86_defs\\[\\] = {$")
|
2020-11-04 11:53:50 +01:00
|
|
|
end_mark = "};\n"
|
|
|
|
shorthand = re.compile("^#define ([A-Z0-9_]+_FEATURES) (.*)$")
|
|
|
|
lines = list()
|
|
|
|
shorthands = dict()
|
|
|
|
|
|
|
|
with open(filename, "rt") as f:
|
|
|
|
while True:
|
|
|
|
line = readline_cont(f)
|
|
|
|
if not line:
|
|
|
|
raise RuntimeError("begin mark not found")
|
2021-06-07 14:01:23 +02:00
|
|
|
match = begin_mark.match(line)
|
|
|
|
if match:
|
2021-06-09 13:04:29 +02:00
|
|
|
break
|
2020-11-04 11:53:50 +01:00
|
|
|
match = shorthand.match(line)
|
|
|
|
if match:
|
|
|
|
# TCG definitions are irrelevant for cpu models
|
|
|
|
newk = match.group(1)
|
|
|
|
if newk.startswith("TCG_"):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# remove comments, whitespace and bit operators, effectively
|
|
|
|
# turning the bitfield into a list
|
|
|
|
newv = re.sub("([()|\t\n])|(/\\*.*?\\*/)", " ", match.group(2))
|
|
|
|
|
|
|
|
# resolve recursive shorthands
|
|
|
|
for k, v in shorthands.items():
|
|
|
|
newv = newv.replace(k, v)
|
|
|
|
|
|
|
|
shorthands[newk] = newv
|
|
|
|
|
|
|
|
while True:
|
|
|
|
line = readline_cont(f)
|
|
|
|
if line == end_mark:
|
|
|
|
break
|
|
|
|
if not line:
|
|
|
|
raise RuntimeError("end marker not found")
|
|
|
|
|
|
|
|
# apply shorthands
|
|
|
|
for k, v in shorthands.items():
|
|
|
|
line = line.replace(k, v)
|
|
|
|
lines.append(line)
|
|
|
|
|
|
|
|
return "".join(lines)
|
|
|
|
|
|
|
|
|
|
|
|
def transform(item):
|
|
|
|
"""Recursively transform a Lark syntax tree into python native objects."""
|
|
|
|
|
|
|
|
if isinstance(item, lark.lexer.Token):
|
|
|
|
return str(item)
|
|
|
|
|
|
|
|
if item.data == "list":
|
|
|
|
retval = list()
|
|
|
|
for child in item.children:
|
|
|
|
value = transform(child)
|
|
|
|
if value is None:
|
|
|
|
continue
|
|
|
|
retval.append(value)
|
|
|
|
return retval
|
|
|
|
|
|
|
|
if item.data == "map":
|
|
|
|
retval = dict()
|
|
|
|
for child in item.children:
|
|
|
|
if len(child.children) != 2:
|
|
|
|
raise RuntimeError("map entry with more than 2 elements")
|
|
|
|
key = transform(child.children[0])
|
|
|
|
value = transform(child.children[1])
|
|
|
|
if key is None:
|
|
|
|
raise RuntimeError("map entry with 'None' key")
|
|
|
|
if value is None:
|
|
|
|
continue
|
|
|
|
retval[key] = value
|
|
|
|
return retval
|
|
|
|
|
|
|
|
if item.data == "text":
|
|
|
|
retval = list()
|
|
|
|
for child in item.children:
|
|
|
|
value = transform(child)
|
|
|
|
if value is None:
|
|
|
|
continue
|
|
|
|
retval.append(value)
|
|
|
|
return " ".join(retval)
|
|
|
|
|
|
|
|
if item.data == "value":
|
|
|
|
if item.children:
|
|
|
|
raise RuntimeError("empty list is not empty")
|
|
|
|
return None
|
|
|
|
|
|
|
|
raise RuntimeError("unexpected item type")
|
|
|
|
|
|
|
|
|
|
|
|
def expand_model(model):
|
|
|
|
"""Expand a qemu cpu model description that has its feature split up into
|
|
|
|
different fields and may have differing versions into several libvirt-
|
|
|
|
friendly cpu models."""
|
|
|
|
|
|
|
|
result = {
|
|
|
|
"name": model.pop(".name"),
|
2020-11-23 15:14:24 +01:00
|
|
|
"vendor": translate_vendor(model.pop(".vendor")),
|
2020-11-04 11:53:50 +01:00
|
|
|
"features": set(),
|
|
|
|
"extra": dict()}
|
|
|
|
|
|
|
|
if ".family" in model and ".model" in model:
|
|
|
|
result["family"] = model.pop(".family")
|
|
|
|
result["model"] = model.pop(".model")
|
|
|
|
|
|
|
|
for k in [k for k in model if k.startswith(".features")]:
|
|
|
|
v = model.pop(k)
|
|
|
|
for feature in v.split():
|
2020-11-23 15:14:25 +01:00
|
|
|
translated = translate_feature(feature)
|
2020-11-04 11:53:50 +01:00
|
|
|
if translated:
|
|
|
|
result["features"].add(translated)
|
|
|
|
|
|
|
|
versions = model.pop(".versions", [])
|
|
|
|
for k, v in model.items():
|
|
|
|
result["extra"]["model" + k] = v
|
|
|
|
yield result
|
|
|
|
|
|
|
|
for version in versions:
|
|
|
|
result = copy.deepcopy(result)
|
|
|
|
result["name"] = version.pop(".alias", result["name"])
|
|
|
|
|
|
|
|
props = version.pop(".props", dict())
|
|
|
|
for k, v in props:
|
2020-11-23 15:14:26 +01:00
|
|
|
if k not in ("model-id", "stepping", "model"):
|
|
|
|
k = translate_feature(k)
|
|
|
|
if k is None:
|
|
|
|
continue
|
|
|
|
|
2020-11-04 11:53:50 +01:00
|
|
|
if v == "on":
|
|
|
|
result["features"].add(k)
|
|
|
|
elif v == "off" and k in result["features"]:
|
|
|
|
result["features"].remove(k)
|
|
|
|
else:
|
|
|
|
result["extra"]["property." + k] = v
|
|
|
|
|
|
|
|
for k, v in version.items():
|
|
|
|
result["extra"]["version" + k] = v
|
|
|
|
|
|
|
|
yield result
|
|
|
|
|
|
|
|
|
|
|
|
def output_model(f, model):
|
|
|
|
if model["extra"]:
|
|
|
|
f.write("<!-- extra info from qemu:\n")
|
|
|
|
for k, v in model["extra"].items():
|
|
|
|
f.write(" '{}': '{}'\n".format(k, v))
|
|
|
|
f.write("-->\n")
|
|
|
|
|
|
|
|
f.write("<cpus>\n")
|
|
|
|
f.write(" <model name='{}'>\n".format(model["name"]))
|
|
|
|
f.write(" <decode host='on' guest='on'/>\n")
|
|
|
|
f.write(" <signature family='{}' model='{}'/>\n".format(
|
|
|
|
model["family"], model["model"]))
|
|
|
|
f.write(" <vendor name='{}'/>\n".format(model["vendor"]))
|
|
|
|
for feature in sorted(model["features"]):
|
|
|
|
f.write(" <feature name='{}'/>\n".format(feature))
|
|
|
|
f.write(" </model>\n")
|
|
|
|
f.write("</cpus>\n")
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description="Synchronize x86 cpu models from QEMU i386 target.")
|
|
|
|
parser.add_argument(
|
|
|
|
"cpufile",
|
|
|
|
help="Path to 'target/i386/cpu.c' file in the QEMU repository",
|
|
|
|
type=os.path.realpath)
|
|
|
|
parser.add_argument(
|
|
|
|
"outdir",
|
|
|
|
help="Path to 'src/cpu_map' directory in the libvirt repository",
|
|
|
|
type=os.path.realpath)
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
builtin_x86_defs = read_builtin_x86_defs(args.cpufile)
|
|
|
|
|
|
|
|
ast = lark.Lark(r"""
|
|
|
|
list: value ( "," value )* ","?
|
|
|
|
map: keyvalue ( "," keyvalue )* ","?
|
|
|
|
keyvalue: IDENTIFIER "=" value
|
|
|
|
?value: text | "{" "}" | "{" list "}" | "{" map "}"
|
|
|
|
text: (IDENTIFIER | "\"" (/[^"]+/)? "\"")+
|
|
|
|
IDENTIFIER: /[\[\]\._&a-zA-Z0-9]/+
|
|
|
|
%ignore (" " | "\r" | "\n" | "\t" | "|" )+
|
|
|
|
%ignore "(" ( "X86CPUVersionDefinition" | "PropValue" ) "[])"
|
|
|
|
%ignore "//" /.*?/ "\n"
|
|
|
|
%ignore "/*" /(.|\n)*?/ "*/"
|
|
|
|
""", start="list").parse(builtin_x86_defs)
|
|
|
|
|
|
|
|
models_json = transform(ast)
|
|
|
|
|
|
|
|
models = list()
|
|
|
|
for model in models_json:
|
|
|
|
models.extend(expand_model(model))
|
|
|
|
|
|
|
|
for model in models:
|
|
|
|
name = os.path.join(args.outdir, "x86_{}.xml".format(model["name"]))
|
|
|
|
with open(name, "wt") as f:
|
|
|
|
output_model(f, model)
|
|
|
|
|
2020-11-23 15:14:29 +01:00
|
|
|
features = set()
|
|
|
|
for model in models:
|
|
|
|
features.update(model["features"])
|
|
|
|
|
|
|
|
try:
|
|
|
|
filename = os.path.join(args.outdir, "x86_features.xml")
|
|
|
|
dom = xml.etree.ElementTree.parse(filename)
|
|
|
|
known = [x.attrib["name"] for x in dom.getroot().iter("feature")]
|
|
|
|
unknown = [x for x in features if x not in known and x is not None]
|
|
|
|
except Exception as e:
|
|
|
|
unknown = []
|
|
|
|
print("warning: Unable to read libvirt x86_features.xml: {}".format(e))
|
|
|
|
|
|
|
|
for x in unknown:
|
|
|
|
print("warning: Feature unknown to libvirt: {}".format(x))
|
|
|
|
|
2020-11-04 11:53:50 +01:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|