ipc: Fix all Flake8 warnings

This commit is contained in:
Ryan Pavlik 2020-05-28 13:09:16 -05:00
parent 4efdb6762a
commit ff3a4d8b50

View file

@ -1,46 +1,52 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
# Copyright 2020, Collabora, Ltd. # Copyright 2020, Collabora, Ltd.
# SPDX-License-Identifier: BSL-1.0 # SPDX-License-Identifier: BSL-1.0
"""Generate code from a JSON file describing the IPC protocol."""
import json import json
import argparse import argparse
class Arg: class Arg:
@classmethod """An IPC call argument."""
def parseArray(cls, a):
ret = []
for elm in a:
ret.append(cls(elm))
return ret
def getFuncArgumentIn(self): @classmethod
if self.isAggregate: def parse_array(cls, a):
"""Turn an array of data into an array of Arg objects."""
return [cls(elm) for elm in a]
def get_func_argument_in(self):
"""Get the type and name of this argument as an input parameter."""
if self.is_aggregate:
return self.typename + " *" + self.name return self.typename + " *" + self.name
else: else:
return self.typename + " " + self.name return self.typename + " " + self.name
def getFuncArgumentOut(self): def get_func_argument_out(self):
"""Get the type and name of this argument as an output parameter."""
return self.typename + " *out_" + self.name return self.typename + " *out_" + self.name
def getStructField(self): def get_struct_field(self):
"""Get the type and name of this argument as a struct field."""
return self.typename + " " + self.name return self.typename + " " + self.name
def dump(self): def dump(self):
"""Dump human-readable output to standard out."""
print("\t\t" + self.typename + ": " + self.name) print("\t\t" + self.typename + ": " + self.name)
def __init__(self, data): def __init__(self, data):
"""Construct an argument."""
self.name = data['name'] self.name = data['name']
self.typename = data['type'] self.typename = data['type']
self.isAggregate = False self.is_aggregate = False
if self.typename.find("struct ") == 0: if self.typename.find("struct ") == 0:
self.isAggregate = True self.is_aggregate = True
if self.typename.find("union ") == 0: if self.typename.find("union ") == 0:
self.isAggregate = True self.is_aggregate = True
def _write_common(f, start, args, indent): def write_with_wrapped_args(f, start, args, indent):
"Write something like a declaration or call." """Write something like a declaration or call."""
f.write("\n" + indent) f.write("\n" + indent)
f.write(start) f.write(start)
# For parameter indenting # For parameter indenting
@ -50,47 +56,53 @@ def _write_common(f, start, args, indent):
def write_decl(f, return_type, function_name, args, indent=""): def write_decl(f, return_type, function_name, args, indent=""):
"""Write a function declaration/definition with wrapped arguments."""
f.write("\n" + indent) f.write("\n" + indent)
f.write(return_type) f.write(return_type)
_write_common(f, write_with_wrapped_args(f,
"{}(".format(function_name), "{}(".format(function_name),
args, args,
indent) indent)
def write_invocation(f, return_val, function_name, args, indent=""): def write_invocation(f, return_val, function_name, args, indent=""):
_write_common(f, """Write a function call with saved return value and wrapped arguments."""
write_with_wrapped_args(f,
"{} = {}(".format(return_val, function_name), "{} = {}(".format(return_val, function_name),
args, args,
indent) indent)
class Call: class Call:
"""A single IPC call."""
def dump(self): def dump(self):
"""Dump human-readable output to standard out."""
print("Call " + self.name) print("Call " + self.name)
if self.inArgs: if self.in_args:
print("\tIn:") print("\tIn:")
for arg in self.inArgs: for arg in self.in_args:
arg.dump() arg.dump()
if self.outArgs: if self.out_args:
print("\tOut:") print("\tOut:")
for arg in self.outArgs: for arg in self.out_args:
arg.dump() arg.dump()
def writeCallDecl(self, f): def write_call_decl(self, f):
"""Write declaration of ipc_call_CALLNAME."""
args = ["struct ipc_connection *ipc_c"] args = ["struct ipc_connection *ipc_c"]
args.extend(arg.getFuncArgumentIn() for arg in self.inArgs) args.extend(arg.get_func_argument_in() for arg in self.in_args)
args.extend(arg.getFuncArgumentOut() for arg in self.outArgs) args.extend(arg.get_func_argument_out() for arg in self.out_args)
if self.outFds: if self.out_fds:
args.extend(("int *fds", "size_t num_fds")) args.extend(("int *fds", "size_t num_fds"))
write_decl(f, 'ipc_result_t', 'ipc_call_' + self.name, args) write_decl(f, 'ipc_result_t', 'ipc_call_' + self.name, args)
def writeHandleDecl(self, f): def write_handle_decl(self, f):
"""Write declaration of ipc_handle_CALLNAME."""
args = ["volatile struct ipc_client_state *cs"] args = ["volatile struct ipc_client_state *cs"]
args.extend(arg.getFuncArgumentIn() for arg in self.inArgs) args.extend(arg.get_func_argument_in() for arg in self.in_args)
args.extend(arg.getFuncArgumentOut() for arg in self.outArgs) args.extend(arg.get_func_argument_out() for arg in self.out_args)
if self.outFds: if self.out_fds:
args.extend(( args.extend((
"size_t max_num_fds", "size_t max_num_fds",
"int *out_fds", "int *out_fds",
@ -98,21 +110,21 @@ class Call:
write_decl(f, 'ipc_result_t', 'ipc_handle_' + self.name, args) write_decl(f, 'ipc_result_t', 'ipc_handle_' + self.name, args)
def __init__(self, name, data): def __init__(self, name, data):
"""Construct a call from call name and call data dictionary."""
self.id = None self.id = None
self.name = name self.name = name
self.inArgs = [] self.in_args = []
self.outArgs = [] self.out_args = []
self.outFds = False self.out_fds = False
for key in data: for key, val in data.items():
if key == 'id': if key == 'id':
self.id = data[key] self.id = val
elif key == 'in': elif key == 'in':
self.inArgs = Arg.parseArray(data[key]) self.in_args = Arg.parse_array(val)
elif key == 'out': elif key == 'out':
self.outArgs = Arg.parseArray(data[key]) self.out_args = Arg.parse_array(val)
elif key == 'out_fds': elif key == 'out_fds':
self.outFds = data[key] self.out_fds = val
else: else:
raise RuntimeError("Unrecognized key") raise RuntimeError("Unrecognized key")
if not self.id: if not self.id:
@ -120,26 +132,27 @@ class Call:
class Proto: class Proto:
"""An IPC protocol containing one or more calls."""
@classmethod @classmethod
def parse(cls, data): def parse(cls, data):
"""Parse a dictionary defining a protocol into Call objects."""
return cls(data) return cls(data)
@classmethod @classmethod
def loadAndParse(cls, file): def load_and_parse(cls, file):
"""Load a JSON file and parse it into Call objects."""
with open(file) as infile: with open(file) as infile:
return cls.parse(json.loads(infile.read())) return cls.parse(json.loads(infile.read()))
def dump(self): def dump(self):
"""Dump human-readable output to standard out."""
for call in self.calls: for call in self.calls:
call.dump() call.dump()
def addCall(self, name, data):
self.calls.append(Call(name, data))
def __init__(self, data): def __init__(self, data):
self.calls = [] """Construct a protocol from a dictionary of calls."""
for name, call in data.items(): self.calls = [Call(name, call) for name, call in data.items()]
self.addCall(name, call)
header = '''// Copyright 2020, Collabora, Ltd. header = '''// Copyright 2020, Collabora, Ltd.
@ -153,7 +166,11 @@ header = '''// Copyright 2020, Collabora, Ltd.
''' '''
def doH(file, p): def generate_h(file, p):
"""Generate protocol header.
Defines command enum, utility functions, and command and reply structures.
"""
f = open(file, "w") f = open(file, "w")
f.write(header.format(brief='Generated IPC protocol header', suffix='')) f.write(header.format(brief='Generated IPC protocol header', suffix=''))
f.write(''' f.write('''
@ -204,27 +221,28 @@ ipc_cmd_to_str(ipc_command_t id)
for call in p.calls: for call in p.calls:
# Should we emit a msg struct. # Should we emit a msg struct.
if call.inArgs: if call.in_args:
f.write("\nstruct ipc_" + call.name + "_msg\n") f.write("\nstruct ipc_" + call.name + "_msg\n")
f.write("{\n") f.write("{\n")
f.write("\tenum ipc_command cmd;\n") f.write("\tenum ipc_command cmd;\n")
for arg in call.inArgs: for arg in call.in_args:
f.write("\t" + arg.getStructField() + ";\n") f.write("\t" + arg.get_struct_field() + ";\n")
f.write("};\n") f.write("};\n")
# Should we emit a reply struct. # Should we emit a reply struct.
if call.outArgs: if call.out_args:
f.write("\nstruct ipc_" + call.name + "_reply\n") f.write("\nstruct ipc_" + call.name + "_reply\n")
f.write("{\n") f.write("{\n")
f.write("\tipc_result_t result;\n") f.write("\tipc_result_t result;\n")
for arg in call.outArgs: for arg in call.out_args:
f.write("\t" + arg.getStructField() + ";\n") f.write("\t" + arg.get_struct_field() + ";\n")
f.write("};\n") f.write("};\n")
f.write("\n// clang-format on\n") f.write("\n// clang-format on\n")
f.close() f.close()
def doClientC(file, p): def generate_client_c(file, p):
"""Generate IPC client proxy source."""
f = open(file, "w") f = open(file, "w")
f.write(header.format(brief='Generated IPC client code', suffix='_client')) f.write(header.format(brief='Generated IPC client code', suffix='_client'))
f.write(''' f.write('''
@ -236,31 +254,31 @@ def doClientC(file, p):
# Loop over all of the calls. # Loop over all of the calls.
for call in p.calls: for call in p.calls:
call.writeCallDecl(f) call.write_call_decl(f)
f.write("\n{\n") f.write("\n{\n")
# Message struct # Message struct
if call.inArgs: if call.in_args:
f.write("\tstruct ipc_" + call.name + "_msg _msg = {\n") f.write("\tstruct ipc_" + call.name + "_msg _msg = {\n")
else: else:
f.write("\tstruct ipc_command_msg _msg = {\n") f.write("\tstruct ipc_command_msg _msg = {\n")
f.write("\t .cmd = " + str(call.id) + ",\n") f.write("\t .cmd = " + str(call.id) + ",\n")
for arg in call.inArgs: for arg in call.in_args:
if arg.isAggregate: if arg.is_aggregate:
f.write("\t ." + arg.name + " = *" + arg.name + ",\n") f.write("\t ." + arg.name + " = *" + arg.name + ",\n")
else: else:
f.write("\t ." + arg.name + " = " + arg.name + ",\n") f.write("\t ." + arg.name + " = " + arg.name + ",\n")
f.write("\t};\n") f.write("\t};\n")
# Reply struct # Reply struct
if call.outArgs: if call.out_args:
f.write("\tstruct ipc_" + call.name + "_reply _reply;\n") f.write("\tstruct ipc_" + call.name + "_reply _reply;\n")
else: else:
f.write("\tstruct ipc_result_reply _reply = {0};\n") f.write("\tstruct ipc_result_reply _reply = {0};\n")
func = 'ipc_client_send_and_get_reply' func = 'ipc_client_send_and_get_reply'
args = ['ipc_c', '&_msg', 'sizeof(_msg)', '&_reply', 'sizeof(_reply)'] args = ['ipc_c', '&_msg', 'sizeof(_msg)', '&_reply', 'sizeof(_reply)']
if call.outFds: if call.out_fds:
func += '_fds' func += '_fds'
args.extend(('fds', 'num_fds')) args.extend(('fds', 'num_fds'))
write_invocation(f, 'ipc_result_t ret', func, args, indent="\t") write_invocation(f, 'ipc_result_t ret', func, args, indent="\t")
@ -270,14 +288,18 @@ def doClientC(file, p):
\t\treturn ret; \t\treturn ret;
\t} \t}
\n''') \n''')
for arg in call.outArgs: for arg in call.out_args:
f.write("\t*out_" + arg.name + " = _reply." + arg.name + ";\n") f.write("\t*out_" + arg.name + " = _reply." + arg.name + ";\n")
f.write("\n\treturn _reply.result;\n}\n") f.write("\n\treturn _reply.result;\n}\n")
f.write("\n// clang-format off\n") f.write("\n// clang-format off\n")
f.close() f.close()
def doClientH(file, p): def generate_client_h(file, p):
"""Generate IPC client header.
Contains prototypes for generated IPC proxy call functions.
"""
f = open(file, "w") f = open(file, "w")
f.write(header.format(brief='Generated IPC client code', suffix='_client')) f.write(header.format(brief='Generated IPC client code', suffix='_client'))
f.write(''' f.write('''
@ -292,13 +314,14 @@ def doClientH(file, p):
''') ''')
for call in p.calls: for call in p.calls:
call.writeCallDecl(f) call.write_call_decl(f)
f.write(";\n") f.write(";\n")
f.write("\n// clang-format on\n") f.write("\n// clang-format on\n")
f.close() f.close()
def doServerC(file, p): def generate_server_c(file, p):
"""Generate IPC server stub/dispatch source."""
f = open(file, "w") f = open(file, "w")
f.write(header.format(brief='Generated IPC server code', suffix='_server')) f.write(header.format(brief='Generated IPC server code', suffix='_server'))
f.write(''' f.write('''
@ -322,38 +345,47 @@ ipc_dispatch(volatile struct ipc_client_state *cs, ipc_command_t *ipc_command)
for call in p.calls: for call in p.calls:
f.write("\tcase " + call.id + ": {\n") f.write("\tcase " + call.id + ": {\n")
if call.inArgs: if call.in_args:
f.write("\t\tstruct ipc_" + call.name + "_msg *msg =\n") f.write(
f.write("\t\t (struct ipc_" + call.name + "_msg *)ipc_command;\n") "\t\tstruct ipc_{}_msg *msg =\n".format(call.name))
if call.outArgs: f.write(
f.write("\t\tstruct ipc_" + call.name + "_reply reply = {0};\n") "\t\t (struct ipc_{}_msg *)ipc_command;\n".format(
call.name))
if call.out_args:
f.write("\t\tstruct ipc_%s_reply reply = {0};\n" % call.name)
else: else:
f.write("\t\tstruct ipc_result_reply reply = {0};\n") f.write("\t\tstruct ipc_result_reply reply = {0};\n")
if call.outFds: if call.out_fds:
f.write("\t\tint fds[MAX_FDS] = {0};\n") f.write("\t\tint fds[MAX_FDS] = {0};\n")
f.write("\t\tsize_t num_fds = {0};\n") f.write("\t\tsize_t num_fds = {0};\n")
f.write("\n") f.write("\n")
# Write call to ipc_handle_CALLNAME # Write call to ipc_handle_CALLNAME
args = ["cs"] args = ["cs"]
for arg in call.inArgs: for arg in call.in_args:
args.append(("&msg->" + arg.name) args.append(("&msg->" + arg.name)
if arg.isAggregate if arg.is_aggregate
else ("msg->" + arg.name)) else ("msg->" + arg.name))
args.extend("&reply." + arg.name for arg in call.outArgs) args.extend("&reply." + arg.name for arg in call.out_args)
if call.outFds: if call.out_fds:
args.extend(("MAX_FDS", args.extend(("MAX_FDS",
"fds", "fds",
"&num_fds",)) "&num_fds",))
write_invocation(f, 'reply.result', 'ipc_handle_' + call.name, args, indent="\t\t") write_invocation(f, 'reply.result', 'ipc_handle_' +
call.name, args, indent="\t\t")
f.write(";\n") f.write(";\n")
if call.outFds: if call.out_fds:
f.write( f.write(
"\t\treturn ipc_reply_fds(cs->ipc_socket_fd, &reply, sizeof(reply), fds, num_fds);\n") "\t\t"
"return ipc_reply_fds(cs->ipc_socket_fd, "
"&reply, sizeof(reply), "
"fds, num_fds);\n")
else: else:
f.write( f.write(
"\t\treturn ipc_reply(cs->ipc_socket_fd, &reply, sizeof(reply));\n") "\t\t"
"return ipc_reply(cs->ipc_socket_fd, "
"&reply, sizeof(reply));\n")
f.write("\t}\n") f.write("\t}\n")
f.write('''\tdefault: f.write('''\tdefault:
\t\tprintf("UNHANDLED IPC MESSAGE! %d\\n", *ipc_command); \t\tprintf("UNHANDLED IPC MESSAGE! %d\\n", *ipc_command);
@ -366,7 +398,12 @@ ipc_dispatch(volatile struct ipc_client_state *cs, ipc_command_t *ipc_command)
f.close() f.close()
def doServerH(file, p): def generate_server_header(file, p):
"""Generate IPC server header.
Declares handler prototypes to implement,
as well as the prototype for the generated dispatch function.
"""
f = open(file, "w") f = open(file, "w")
f.write(header.format(brief='Generated IPC server code', suffix='_server')) f.write(header.format(brief='Generated IPC server code', suffix='_server'))
f.write(''' f.write('''
@ -383,32 +420,35 @@ ipc_dispatch(volatile struct ipc_client_state *cs, ipc_command_t *ipc_command);
''') ''')
for call in p.calls: for call in p.calls:
call.writeHandleDecl(f) call.write_handle_decl(f)
f.write(";\n") f.write(";\n")
f.write("\n// clang-format on\n") f.write("\n// clang-format on\n")
f.close() f.close()
def main(): def main():
"""Handle command line and generate a file."""
parser = argparse.ArgumentParser(description='Protocol generator.') parser = argparse.ArgumentParser(description='Protocol generator.')
parser.add_argument('proto', help='Protocol file to use') parser.add_argument(
parser.add_argument('output', type=str, nargs='+', 'proto', help='Protocol file to use')
help='Output file, uses the ending to figure out what file it should generate') parser.add_argument(
'output', type=str, nargs='+',
help='Output file, uses the name to choose output type')
args = parser.parse_args() args = parser.parse_args()
p = Proto.loadAndParse(args.proto) p = Proto.load_and_parse(args.proto)
for output in args.output: for output in args.output:
if output.endswith("ipc_protocol_generated.h"): if output.endswith("ipc_protocol_generated.h"):
doH(output, p) generate_h(output, p)
if output.endswith("ipc_client_generated.c"): if output.endswith("ipc_client_generated.c"):
doClientC(output, p) generate_client_c(output, p)
if output.endswith("ipc_client_generated.h"): if output.endswith("ipc_client_generated.h"):
doClientH(output, p) generate_client_h(output, p)
if output.endswith("ipc_server_generated.c"): if output.endswith("ipc_server_generated.c"):
doServerC(output, p) generate_server_c(output, p)
if output.endswith("ipc_server_generated.h"): if output.endswith("ipc_server_generated.h"):
doServerH(output, p) generate_server_header(output, p)
if __name__ == "__main__": if __name__ == "__main__":