1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright 2012 The WebRTC Project Authors. All rights reserved.
3*d9f75844SAndroid Build Coastguard Worker *
4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker */
10*d9f75844SAndroid Build Coastguard Worker
11*d9f75844SAndroid Build Coastguard Worker // clang-format off
12*d9f75844SAndroid Build Coastguard Worker // clang formating would change include order.
13*d9f75844SAndroid Build Coastguard Worker #include <windows.h>
14*d9f75844SAndroid Build Coastguard Worker #include <shellapi.h> // must come after windows.h
15*d9f75844SAndroid Build Coastguard Worker // clang-format on
16*d9f75844SAndroid Build Coastguard Worker
17*d9f75844SAndroid Build Coastguard Worker #include <string>
18*d9f75844SAndroid Build Coastguard Worker #include <vector>
19*d9f75844SAndroid Build Coastguard Worker
20*d9f75844SAndroid Build Coastguard Worker #include "absl/flags/parse.h"
21*d9f75844SAndroid Build Coastguard Worker #include "examples/peerconnection/client/conductor.h"
22*d9f75844SAndroid Build Coastguard Worker #include "examples/peerconnection/client/flag_defs.h"
23*d9f75844SAndroid Build Coastguard Worker #include "examples/peerconnection/client/main_wnd.h"
24*d9f75844SAndroid Build Coastguard Worker #include "examples/peerconnection/client/peer_connection_client.h"
25*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
26*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/ssl_adapter.h"
27*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/string_utils.h" // For ToUtf8
28*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/win32_socket_init.h"
29*d9f75844SAndroid Build Coastguard Worker #include "system_wrappers/include/field_trial.h"
30*d9f75844SAndroid Build Coastguard Worker #include "test/field_trial.h"
31*d9f75844SAndroid Build Coastguard Worker
32*d9f75844SAndroid Build Coastguard Worker namespace {
33*d9f75844SAndroid Build Coastguard Worker // A helper class to translate Windows command line arguments into UTF8,
34*d9f75844SAndroid Build Coastguard Worker // which then allows us to just pass them to the flags system.
35*d9f75844SAndroid Build Coastguard Worker // This encapsulates all the work of getting the command line and translating
36*d9f75844SAndroid Build Coastguard Worker // it to an array of 8-bit strings; all you have to do is create one of these,
37*d9f75844SAndroid Build Coastguard Worker // and then call argc() and argv().
38*d9f75844SAndroid Build Coastguard Worker class WindowsCommandLineArguments {
39*d9f75844SAndroid Build Coastguard Worker public:
40*d9f75844SAndroid Build Coastguard Worker WindowsCommandLineArguments();
41*d9f75844SAndroid Build Coastguard Worker
42*d9f75844SAndroid Build Coastguard Worker WindowsCommandLineArguments(const WindowsCommandLineArguments&) = delete;
43*d9f75844SAndroid Build Coastguard Worker WindowsCommandLineArguments& operator=(WindowsCommandLineArguments&) = delete;
44*d9f75844SAndroid Build Coastguard Worker
argc()45*d9f75844SAndroid Build Coastguard Worker int argc() { return argv_.size(); }
argv()46*d9f75844SAndroid Build Coastguard Worker char** argv() { return argv_.data(); }
47*d9f75844SAndroid Build Coastguard Worker
48*d9f75844SAndroid Build Coastguard Worker private:
49*d9f75844SAndroid Build Coastguard Worker // Owned argument strings.
50*d9f75844SAndroid Build Coastguard Worker std::vector<std::string> args_;
51*d9f75844SAndroid Build Coastguard Worker // Pointers, to get layout compatible with char** argv.
52*d9f75844SAndroid Build Coastguard Worker std::vector<char*> argv_;
53*d9f75844SAndroid Build Coastguard Worker };
54*d9f75844SAndroid Build Coastguard Worker
WindowsCommandLineArguments()55*d9f75844SAndroid Build Coastguard Worker WindowsCommandLineArguments::WindowsCommandLineArguments() {
56*d9f75844SAndroid Build Coastguard Worker // start by getting the command line.
57*d9f75844SAndroid Build Coastguard Worker LPCWSTR command_line = ::GetCommandLineW();
58*d9f75844SAndroid Build Coastguard Worker // now, convert it to a list of wide char strings.
59*d9f75844SAndroid Build Coastguard Worker int argc;
60*d9f75844SAndroid Build Coastguard Worker LPWSTR* wide_argv = ::CommandLineToArgvW(command_line, &argc);
61*d9f75844SAndroid Build Coastguard Worker
62*d9f75844SAndroid Build Coastguard Worker // iterate over the returned wide strings;
63*d9f75844SAndroid Build Coastguard Worker for (int i = 0; i < argc; ++i) {
64*d9f75844SAndroid Build Coastguard Worker args_.push_back(rtc::ToUtf8(wide_argv[i], wcslen(wide_argv[i])));
65*d9f75844SAndroid Build Coastguard Worker // make sure the argv array points to the string data.
66*d9f75844SAndroid Build Coastguard Worker argv_.push_back(const_cast<char*>(args_.back().c_str()));
67*d9f75844SAndroid Build Coastguard Worker }
68*d9f75844SAndroid Build Coastguard Worker LocalFree(wide_argv);
69*d9f75844SAndroid Build Coastguard Worker }
70*d9f75844SAndroid Build Coastguard Worker
71*d9f75844SAndroid Build Coastguard Worker } // namespace
wWinMain(HINSTANCE instance,HINSTANCE prev_instance,wchar_t * cmd_line,int cmd_show)72*d9f75844SAndroid Build Coastguard Worker int PASCAL wWinMain(HINSTANCE instance,
73*d9f75844SAndroid Build Coastguard Worker HINSTANCE prev_instance,
74*d9f75844SAndroid Build Coastguard Worker wchar_t* cmd_line,
75*d9f75844SAndroid Build Coastguard Worker int cmd_show) {
76*d9f75844SAndroid Build Coastguard Worker rtc::WinsockInitializer winsock_init;
77*d9f75844SAndroid Build Coastguard Worker rtc::PhysicalSocketServer ss;
78*d9f75844SAndroid Build Coastguard Worker rtc::AutoSocketServerThread main_thread(&ss);
79*d9f75844SAndroid Build Coastguard Worker
80*d9f75844SAndroid Build Coastguard Worker WindowsCommandLineArguments win_args;
81*d9f75844SAndroid Build Coastguard Worker int argc = win_args.argc();
82*d9f75844SAndroid Build Coastguard Worker char** argv = win_args.argv();
83*d9f75844SAndroid Build Coastguard Worker
84*d9f75844SAndroid Build Coastguard Worker absl::ParseCommandLine(argc, argv);
85*d9f75844SAndroid Build Coastguard Worker
86*d9f75844SAndroid Build Coastguard Worker // InitFieldTrialsFromString stores the char*, so the char array must outlive
87*d9f75844SAndroid Build Coastguard Worker // the application.
88*d9f75844SAndroid Build Coastguard Worker const std::string forced_field_trials =
89*d9f75844SAndroid Build Coastguard Worker absl::GetFlag(FLAGS_force_fieldtrials);
90*d9f75844SAndroid Build Coastguard Worker webrtc::field_trial::InitFieldTrialsFromString(forced_field_trials.c_str());
91*d9f75844SAndroid Build Coastguard Worker
92*d9f75844SAndroid Build Coastguard Worker // Abort if the user specifies a port that is outside the allowed
93*d9f75844SAndroid Build Coastguard Worker // range [1, 65535].
94*d9f75844SAndroid Build Coastguard Worker if ((absl::GetFlag(FLAGS_port) < 1) || (absl::GetFlag(FLAGS_port) > 65535)) {
95*d9f75844SAndroid Build Coastguard Worker printf("Error: %i is not a valid port.\n", absl::GetFlag(FLAGS_port));
96*d9f75844SAndroid Build Coastguard Worker return -1;
97*d9f75844SAndroid Build Coastguard Worker }
98*d9f75844SAndroid Build Coastguard Worker
99*d9f75844SAndroid Build Coastguard Worker const std::string server = absl::GetFlag(FLAGS_server);
100*d9f75844SAndroid Build Coastguard Worker MainWnd wnd(server.c_str(), absl::GetFlag(FLAGS_port),
101*d9f75844SAndroid Build Coastguard Worker absl::GetFlag(FLAGS_autoconnect), absl::GetFlag(FLAGS_autocall));
102*d9f75844SAndroid Build Coastguard Worker if (!wnd.Create()) {
103*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_NOTREACHED();
104*d9f75844SAndroid Build Coastguard Worker return -1;
105*d9f75844SAndroid Build Coastguard Worker }
106*d9f75844SAndroid Build Coastguard Worker
107*d9f75844SAndroid Build Coastguard Worker rtc::InitializeSSL();
108*d9f75844SAndroid Build Coastguard Worker PeerConnectionClient client;
109*d9f75844SAndroid Build Coastguard Worker auto conductor = rtc::make_ref_counted<Conductor>(&client, &wnd);
110*d9f75844SAndroid Build Coastguard Worker
111*d9f75844SAndroid Build Coastguard Worker // Main loop.
112*d9f75844SAndroid Build Coastguard Worker MSG msg;
113*d9f75844SAndroid Build Coastguard Worker BOOL gm;
114*d9f75844SAndroid Build Coastguard Worker while ((gm = ::GetMessage(&msg, NULL, 0, 0)) != 0 && gm != -1) {
115*d9f75844SAndroid Build Coastguard Worker if (!wnd.PreTranslateMessage(&msg)) {
116*d9f75844SAndroid Build Coastguard Worker ::TranslateMessage(&msg);
117*d9f75844SAndroid Build Coastguard Worker ::DispatchMessage(&msg);
118*d9f75844SAndroid Build Coastguard Worker }
119*d9f75844SAndroid Build Coastguard Worker }
120*d9f75844SAndroid Build Coastguard Worker
121*d9f75844SAndroid Build Coastguard Worker if (conductor->connection_active() || client.is_connected()) {
122*d9f75844SAndroid Build Coastguard Worker while ((conductor->connection_active() || client.is_connected()) &&
123*d9f75844SAndroid Build Coastguard Worker (gm = ::GetMessage(&msg, NULL, 0, 0)) != 0 && gm != -1) {
124*d9f75844SAndroid Build Coastguard Worker if (!wnd.PreTranslateMessage(&msg)) {
125*d9f75844SAndroid Build Coastguard Worker ::TranslateMessage(&msg);
126*d9f75844SAndroid Build Coastguard Worker ::DispatchMessage(&msg);
127*d9f75844SAndroid Build Coastguard Worker }
128*d9f75844SAndroid Build Coastguard Worker }
129*d9f75844SAndroid Build Coastguard Worker }
130*d9f75844SAndroid Build Coastguard Worker
131*d9f75844SAndroid Build Coastguard Worker rtc::CleanupSSL();
132*d9f75844SAndroid Build Coastguard Worker return 0;
133*d9f75844SAndroid Build Coastguard Worker }
134