blob: 2c6a84dfcf2e76813e0d6c5d90aa76344acc2c56 [file] [log] [blame]
Alex Light39795712017-12-14 11:58:21 -08001#!/usr/bin/env python3
2#
3# Copyright 2017, The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17"""
18A python program that simulates the plugin side of the dt_fd_forward transport for testing.
19
20This program will invoke a given java language runtime program and send down debugging arguments
21that cause it to use the dt_fd_forward transport. This will then create a normal server-port that
22debuggers can attach to.
23"""
24
25import argparse
26import array
27from multiprocessing import Process
28import contextlib
29import ctypes
30import os
31import select
32import socket
33import subprocess
34import sys
35import time
36
Alex Light15b81132018-01-24 13:29:07 -080037NEED_HANDSHAKE_MESSAGE = b"HANDSHAKE:REQD\x00"
38LISTEN_START_MESSAGE = b"dt_fd_forward:START-LISTEN\x00"
39LISTEN_END_MESSAGE = b"dt_fd_forward:END-LISTEN\x00"
40ACCEPTED_MESSAGE = b"dt_fd_forward:ACCEPTED\x00"
Alex Lightf51d1822021-02-01 19:25:35 -080041HANDSHAKEN_MESSAGE = b"dt_fd_forward:HANDSHAKE-COMPLETE\x00"
Alex Light15b81132018-01-24 13:29:07 -080042CLOSE_MESSAGE = b"dt_fd_forward:CLOSING\x00"
Alex Light39795712017-12-14 11:58:21 -080043
44libc = ctypes.cdll.LoadLibrary("libc.so.6")
45def eventfd(init_val, flags):
46 """
47 Creates an eventfd. See 'man 2 eventfd' for more information.
48 """
49 return libc.eventfd(init_val, flags)
50
51@contextlib.contextmanager
52def make_eventfd(init):
53 """
54 Creates an eventfd with given initial value that is closed after the manager finishes.
55 """
56 fd = eventfd(init, 0)
57 yield fd
58 os.close(fd)
59
60@contextlib.contextmanager
61def make_sockets():
62 """
63 Make a (remote,local) socket pair. The remote socket is inheritable by forked processes. They are
64 both linked together.
65 """
66 (rfd, lfd) = socket.socketpair(socket.AF_UNIX, socket.SOCK_SEQPACKET)
67 yield (rfd, lfd)
68 rfd.close()
69 lfd.close()
70
71def send_fds(sock, remote_read, remote_write, remote_event):
72 """
73 Send the three fds over the given socket.
74 """
Alex Light15b81132018-01-24 13:29:07 -080075 sock.sendmsg([NEED_HANDSHAKE_MESSAGE], # We want the transport to handle the handshake.
Alex Light39795712017-12-14 11:58:21 -080076 [(socket.SOL_SOCKET, # Send over socket.
77 socket.SCM_RIGHTS, # Payload is file-descriptor array
78 array.array('i', [remote_read, remote_write, remote_event]))])
79
80
81def HandleSockets(host, port, local_sock, finish_event):
82 """
83 Handle the IO between the network and the runtime.
84
85 This is similar to what we will do with the plugin that controls the jdwp connection.
86
87 The main difference is it will keep around the connection and event-fd in order to let it send
88 ddms packets directly.
89 """
90 listening = False
91 with socket.socket() as sock:
92 sock.bind((host, port))
93 sock.listen()
94 while True:
95 sources = [local_sock, finish_event, sock]
96 print("Starting select on " + str(sources))
97 (rf, _, _) = select.select(sources, [], [])
98 if local_sock in rf:
99 buf = local_sock.recv(1024)
100 print("Local_sock has data: " + str(buf))
101 if buf == LISTEN_START_MESSAGE:
102 print("listening on " + str(sock))
103 listening = True
104 elif buf == LISTEN_END_MESSAGE:
105 print("End listening")
106 listening = False
107 elif buf == ACCEPTED_MESSAGE:
108 print("Fds were accepted.")
Alex Lightf51d1822021-02-01 19:25:35 -0800109 elif buf == HANDSHAKEN_MESSAGE:
110 print("Handshake completed.")
Alex Light39795712017-12-14 11:58:21 -0800111 elif buf == CLOSE_MESSAGE:
112 # TODO Dup the fds and send a fake DDMS message like the actual plugin would.
113 print("Fds were closed")
114 else:
115 print("Unknown data received from socket " + str(buf))
116 return
117 elif sock in rf:
118 (conn, addr) = sock.accept()
119 with conn:
120 print("connection accepted from " + str(addr))
121 if listening:
122 with make_eventfd(1) as efd:
123 print("sending fds ({}, {}, {}) to target.".format(conn.fileno(), conn.fileno(), efd))
124 send_fds(local_sock, conn.fileno(), conn.fileno(), efd)
125 else:
126 print("Closing fds since we cannot accept them.")
127 if finish_event in rf:
128 print("woke up from finish_event")
129 return
130
131def StartChildProcess(cmd_pre, cmd_post, jdwp_lib, jdwp_ops, remote_sock, can_be_runtest):
132 """
133 Open the child java-language runtime process.
134 """
135 full_cmd = list(cmd_pre)
136 os.set_inheritable(remote_sock.fileno(), True)
137 jdwp_arg = jdwp_lib + "=" + \
138 jdwp_ops + "transport=dt_fd_forward,address=" + str(remote_sock.fileno())
139 if can_be_runtest and cmd_pre[0].endswith("run-test"):
140 print("Assuming run-test. Pass --no-run-test if this isn't true")
141 full_cmd += ["--with-agent", jdwp_arg]
142 else:
143 full_cmd.append("-agentpath:" + jdwp_arg)
144 full_cmd += cmd_post
145 print("Running " + str(full_cmd))
146 # Start the actual process with the fd being passed down.
147 proc = subprocess.Popen(full_cmd, close_fds=False)
148 # Get rid of the extra socket.
149 remote_sock.close()
150 proc.wait()
151
152def main():
153 parser = argparse.ArgumentParser(description="""
154 Runs a socket that forwards to dt_fds.
155
156 Pass '--' to start passing in the program we will pass the debug
157 options down to.
158 """)
159 parser.add_argument("--host", type=str, default="localhost",
160 help="Host we will listen for traffic on. Defaults to 'localhost'.")
161 parser.add_argument("--debug-lib", type=str, default="libjdwp.so",
162 help="jdwp library we pass to -agentpath:. Default is 'libjdwp.so'")
163 parser.add_argument("--debug-options", type=str, default="server=y,suspend=y,",
164 help="non-address options we pass to jdwp agent, default is " +
165 "'server=y,suspend=y,'")
166 parser.add_argument("--port", type=int, default=12345,
167 help="port we will expose the traffic on. Defaults to 12345.")
168 parser.add_argument("--no-run-test", default=False, action="store_true",
169 help="don't pass in arguments for run-test even if it looks like that is " +
170 "the program")
171 parser.add_argument("--pre-end", type=int, default=1,
172 help="number of 'rest' arguments to put before passing in the debug options")
173 end_idx = 0 if '--' not in sys.argv else sys.argv.index('--')
174 if end_idx == 0 and ('--help' in sys.argv or '-h' in sys.argv):
175 parser.print_help()
176 return
177 args = parser.parse_args(sys.argv[:end_idx][1:])
178 rest = sys.argv[1 + end_idx:]
179
180 with make_eventfd(0) as wakeup_event:
181 with make_sockets() as (remote_sock, local_sock):
182 invoker = Process(target=StartChildProcess,
183 args=(rest[:args.pre_end],
184 rest[args.pre_end:],
185 args.debug_lib,
186 args.debug_options,
187 remote_sock,
188 not args.no_run_test))
189 socket_handler = Process(target=HandleSockets,
190 args=(args.host, args.port, local_sock, wakeup_event))
191 socket_handler.start()
192 invoker.start()
193 invoker.join()
194 # Write any 64 bit value to the wakeup_event to make sure that the socket handler will wake
195 # up and exit.
196 os.write(wakeup_event, b'\x00\x00\x00\x00\x00\x00\x01\x00')
197 socket_handler.join()
198
199if __name__ == '__main__':
200 main()