summaryrefslogtreecommitdiff
path: root/kvmd/apps/edidconf
diff options
context:
space:
mode:
authorMaxim Devaev <[email protected]>2022-06-09 07:12:19 +0300
committerMaxim Devaev <[email protected]>2022-06-09 07:12:19 +0300
commit9ed0c9add9138a1c05319677bb12506b6bf23095 (patch)
tree51d7285c103def8cdb4d9a5358638c935540cc54 /kvmd/apps/edidconf
parente5d5d7938b58585410799c603e90f629dc24b2c4 (diff)
improved kvmd-edidconf
Diffstat (limited to 'kvmd/apps/edidconf')
-rw-r--r--kvmd/apps/edidconf/__init__.py137
1 files changed, 82 insertions, 55 deletions
diff --git a/kvmd/apps/edidconf/__init__.py b/kvmd/apps/edidconf/__init__.py
index 3ef3e7f4..ba411fed 100644
--- a/kvmd/apps/edidconf/__init__.py
+++ b/kvmd/apps/edidconf/__init__.py
@@ -21,27 +21,53 @@
import sys
+import os
import re
+import contextlib
import argparse
from typing import List
+from typing import IO
+from typing import Generator
from typing import Optional
from ...validators.basic import valid_bool
from ...validators.basic import valid_int_f0
-from .. import init
+# from .. import init
# =====
+def _smart_open(path: str, mode: str) -> Generator[IO, None, None]:
+ fd = (0 if "r" in mode else 1)
+ with (os.fdopen(fd, mode, closefd=False) if path == "-" else open(path, mode)) as file:
+ yield file
+ if "w" in mode:
+ file.flush()
+
+
class _Edid:
# https://en.wikipedia.org/wiki/Extended_Display_Identification_Data
def __init__(self, path: str) -> None:
- with open(path) as file:
- self.__load_from_hex(file.read())
+ with _smart_open(path, "rb") as file:
+ data = file.read()
+ if data.startswith(b"\x00\xFF\xFF\xFF\xFF\xFF\xFF\x00"):
+ self.__data = list(data)
+ else:
+ text = re.sub(r"\s", "", data.decode())
+ self.__data = [
+ int(text[index:index + 2], 16)
+ for index in range(0, len(text), 2)
+ ]
+ assert len(self.__data) == 256, f"Invalid EDID length: {len(self.__data)}, should be 256 bytes"
+ assert self.__data[126] == 1, "Zero extensions number"
+ assert (self.__data[128], self.__data[129]) == (0x02, 0x03), "Can't find CEA-861"
+
+ # =====
- def write(self, path: str) -> None:
+ def write_hex(self, path: str) -> None:
self.__update_checksums()
text = "\n".join(
"".join(
@@ -49,12 +75,17 @@ class _Edid:
for item in self.__data[index:index + 16]
)
for index in range(0, len(self.__data), 16)
- )
- if path:
- with open(path, "w") as file:
- file.write(text + "\n")
- else:
- print(text)
+ ) + "\n"
+ with _smart_open(path, "w") as file:
+ file.write(text)
+
+ def write_bin(self, path: str) -> None:
+ with _smart_open(path, "wb") as file:
+ file.write(bytes(self.__data))
+
+ def __update_checksums(self) -> None:
+ self.__data[127] = 256 - (sum(self.__data[:127]) % 256)
+ self.__data[255] = 256 - (sum(self.__data[128:255]) % 256)
# =====
@@ -109,17 +140,21 @@ class _Edid:
# =====
def get_monitor_name(self) -> str:
- index = self.__find_dtd_value(0xFC)
- assert index > 0, "Can't find DTD Monitor name"
+ index = self.__find_mnd_text()
return bytes(self.__data[index:index + 13]).decode("cp437").strip()
def set_monitor_name(self, name: str) -> None:
- index = self.__find_dtd_value(0xFC)
- assert index > 0, "Can't find DTD Monitor name"
+ index = self.__find_mnd_text()
encoded = (name[:13] + "\n" + " " * 12)[:13].encode("cp437")
for (offset, byte) in enumerate(encoded):
self.__data[index + offset] = byte
+ def __find_mnd_text(self) -> int:
+ for index in [54, 72, 90, 108]:
+ if self.__data[index + 3] == 0xFC:
+ return index + 5
+ raise AssertionError("Can't find DTD Monitor name")
+
# =====
def get_audio(self) -> bool:
@@ -131,44 +166,28 @@ class _Edid:
else:
self.__data[131] &= (0xFF - 0b01000000) # ~X
- # =====
-
- def __find_dtd_value(self, dtype: int) -> int:
- for index in [54, 72, 90, 108]:
- if self.__data[index + 3] == dtype:
- return index + 5
- return -1
-
- def __load_from_hex(self, text: str) -> None:
- text = re.sub(r"\s", "", text)
- self.__data = [
- int(text[index:index + 2], 16)
- for index in range(0, len(text), 2)
- ]
- assert len(self.__data) == 256, f"Invalid EDID length: {len(self.__data)}, should be 256 bytes"
- assert self.__data[126] == 1, "Zero extensions number"
- assert (self.__data[128], self.__data[129]) == (0x02, 0x03), "Can't find CEA-861"
-
- def __update_checksums(self) -> None:
- self.__data[127] = 256 - (sum(self.__data[:127]) % 256)
- self.__data[255] = 256 - (sum(self.__data[128:255]) % 256)
-
# =====
def main(argv: Optional[List[str]]=None) -> None:
- (parent_parser, argv, _) = init(
- add_help=False,
- argv=argv,
- )
+ # (parent_parser, argv, _) = init(
+ # add_help=False,
+ # argv=argv,
+ # )
+ if argv is None:
+ argv = sys.argv
parser = argparse.ArgumentParser(
prog="kvmd-edidconf",
description="A simple and primitive KVMD EDID editor",
- parents=[parent_parser],
+ # parents=[parent_parser],
)
parser.add_argument("-f", "--edid-file", dest="path", default="/etc/kvmd/tc358743-edid.hex",
- help="EDID hex text file path", metavar="<file>")
- parser.add_argument("--stdout", action="store_true",
- help="Write to stdout instead of the rewriting the source file")
+ help="The hex/bin EDID file path", metavar="<file>")
+ parser.add_argument("--export-hex", default=None,
+ help="Export [--edid-file] to the new file as a hex text", metavar="<file>")
+ parser.add_argument("--export-bin", default=None,
+ help="Export [--edid-file] to the new file as a bin data", metavar="<file>")
+ parser.add_argument("--import", default=None, dest="imp",
+ help="Import specified bin/hex EDID to the [--edid-file] as a hex text", metavar="<file>")
parser.add_argument("--set-audio", type=valid_bool, default=None,
help="Enable or disable basic audio", metavar="<yes|no>")
parser.add_argument("--set-mfc-id", default=None,
@@ -179,10 +198,12 @@ def main(argv: Optional[List[str]]=None) -> None:
help="Set serial number (decimal)", metavar="<uint>")
parser.add_argument("--set-monitor-name", default=None,
help="Set monitor name in DTD/MND (ASCII, max 13 characters)", metavar="<str>")
- parser.add_argument("--show-info", action="store_true",
- help="Write summary info to stderr")
options = parser.parse_args(argv[1:])
+ if options.imp:
+ options.export_hex = options.path
+ options.path = options.imp
+
edid = _Edid(options.path)
changed = False
@@ -193,12 +214,18 @@ def main(argv: Optional[List[str]]=None) -> None:
getattr(edid, cmd)(value)
changed = True
- if changed:
- edid.write("" if options.stdout else options.path)
-
- if options.show_info:
- print("Manufacturer ID:", edid.get_mfc_id())
- print("Product ID: ", edid.get_product_id())
- print("Serial number: ", edid.get_serial())
- print("Monitor name: ", edid.get_monitor_name())
- print("Basic audio: ", ("yes" if edid.get_audio() else "no"), file=sys.stderr)
+ if options.export_hex is not None:
+ edid.write_hex(options.export_hex)
+ elif options.export_bin is not None:
+ edid.write_bin(options.export_bin)
+ elif changed:
+ edid.write_hex(options.path)
+
+ for (key, value) in [
+ ("Manufacturer ID:", edid.get_mfc_id()),
+ ("Product ID: ", edid.get_product_id()),
+ ("Serial number: ", edid.get_serial()),
+ ("Monitor name: ", edid.get_monitor_name()),
+ ("Basic audio: ", ("yes" if edid.get_audio() else "no")),
+ ]:
+ print(key, value, file=sys.stderr)