page_type | urlFragment | languages | products | description | ||
---|---|---|---|---|---|---|
sample |
external-process-launcher |
|
|
We'll learn how to launch an external process (exe) from a Universal Windows Platform (UWP) app using the Windows.System.ProcessLauncher API with Windows 10 IoT Core. |
We'll learn how to launch an external process (exe) from a Universal Windows Platform (UWP) app using Windows.System.ProcessLauncher
API.
This is a headed sample. To better understand what headed mode is and how to configure your device to be headed, follow the instructions here.
The Windows.System.ProcessLauncher
API is new to Windows IoT Core Fall Update. You can download a Windows 10 IoT Core image with Fall Update from our downloads page.
To use the ProcessLauncher
API and other new features of Windows IoT Core Fall Update, a newer version of the Windows SDK is needed as well. Windows SDK 10.0.10586.0 or higher is required and can be downloaded from here.
You can find the source code for this sample by downloading a zip of all of our samples here.
Make a copy of the folder on your disk and open the project from Visual Studio.
The code for the ProcessLauncher sample can be found under: \ProcessLauncher\CS
If you're building for Minnowboard Max, select x86
as the architecture. If you're building for Raspberry Pi 2 or 3 or DragonBoard , select ARM
.
Select Remote Machine to point to IoT device and hit F5 to deploy to your device. Go back to the basic 'Hello World' sample if you need guidance.
The sample app when deployed displays a screen similar to this one:
Go ahead, click the Run Command button to launch the application specified in the Command edit box, SampleConsoleApplication.exe, which is an executable Win32 application built and deployed from the sample app as well.
When run, the SampleConsoleApplication exe, will send output to both the Standard Output and Standard Error boxes. The return error code of the process, 100, should also be shown.
The code for this sample makes use of the Windows.System.ProcessLauncher
API, which is part of the system management contract APIs available on Windows IoT Core.
To use the API, we need to do a couple of things.
Since the IoT extension SDK is not added to projects by default, we'll need to add a reference to it so its types, including Windows.System.SystemManagement.ProcessLauncher
, will be available in the project.
To do so, right-click on the References entry under the project, select "Add Reference" then navigate the resulting dialog to Universal Windows->Extensions->Windows IoT Extensions for the UWP
making sure the right version, 10.0.10586.0 or higher, is selected, then check the box, and click OK.
Accessing the Windows.System.ProcessLauncher
API from your code requires the systemManagement
IOT Capability. So, we need to add it to the AppX manifest, Package.appxmanifest file:
NOTE: While you can add other capabilities directly by double clicking and opening the Package.appxmanifest file in the UI editor, systemManagement capability can only be added via the XML editor (Right Click on the file -> Open with -> XML (Text) Editor) and adding the capability below:
<Package
xmlns="http://schemas.microsoft.com/appx/manifest/foundation/windows10"
xmlns:mp="http://schemas.microsoft.com/appx/2014/phone/manifest"
xmlns:uap="http://schemas.microsoft.com/appx/manifest/uap/windows10"
xmlns:iot="http://schemas.microsoft.com/appx/manifest/iot/windows10"
IgnorableNamespaces="uap mp iot">
<Capabilities>
<Capability Name="internetClient" />
<!-- Add the capability here -->
<iot:Capability Name="systemManagement" />
</Capabilities>
The ProcessLauncher
API providers a number of static method overloads to launch new apps using the executable file name plus optional arguments and other options.
Here's the part of the sample that launches the new process. After the process is launched, we await
its return to get the exit code.
var result = await ProcessLauncher.RunToCompletionAsync(cmd.Text, args.Text == null ? string.Empty : args.Text, options);
ProcessExitCode.Text += "Process Exit Code: " + result.ExitCode;
While the exit code is often enough to let us know if an executable has succeeded or not, sometimes we need to read the output from the program; e.g. to log information or to show that text in the app.
The ProcessLauncherOptions
class allows sharing the standard input, output and error streams between the UWP app and external process.
ProcessLauncherOptions.StandardInput
is of typeWindows.Storage.Streams.IInputStream
and is read from by the external process- Both
ProcessLauncherOptions.StandardOutput
andProcessLauncherOptions.StandardError
areWindows.Storage.Streams.IInputStream
properties and are used by the external process to write to standard output or standard error
Now, let's learn how to pass output streams to the executable and then read what was written to it.
To be able to both read and write data to strams, we create the streams as Windows.Storage.Streams.InMemoryRandomAccessStream
objects. InMemoryRandomAccessStream
can be casted as both Input or Output streams, which is exactly what we need. We will pass it as IOutputStream
to the ProcessLauncherOptions
, and then read data from it as an IInputStream
after the external exe has completed writing to it.
Here's the relavant code from the sample. First, we initialize the stream objects:
var options = new ProcessLauncherOptions();
var standardOutput = new InMemoryRandomAccessStream();
var standardError = new InMemoryRandomAccessStream();
options.StandardOutput = standardOutput;
options.StandardError = standardError;
Then, we pass the options to the RunToCompletionAsync()
method:
var result = await ProcessLauncher.RunToCompletionAsync(cmd.Text, args.Text == null ? string.Empty : args.Text, options);
Finally, we read the data from the InMemoryRandomAccessStream
after getting it as an input stream:
using (var outStreamRedirect = standardOutput.GetInputStreamAt(0))
{
var size = standardOutput.Size;
using (var dataReader = new DataReader(outStreamRedirect))
{
var bytesLoaded = await dataReader.LoadAsync((uint)size);
var stringRead = dataReader.ReadString(bytesLoaded);
StdOutputText.Text += stringRead;
}
}
Note, while not used in the sample, the standard input stream can be used with the ProcessLauncher
API as well.
The ProcessLauncher
API can launch executable programs packaged as part of the same AppX. So, a sample Win32 console app, SampleConsoleApplication, is included in the sample.
The program itself is quite simple, but it demonstrates the various aspects of the ProcessLauncher API, including argument passing and sharing standard output and standard error streams.
Here's the main method of the sample console app:
int main(int argc, char **argv)
{
std::cout << "Hi there!" << std::endl;
std::cout << "I'm running in a different process!" << std::endl << std::endl;
std::cout << "There are " << argc << " arguments passed to this application (including the exe name):" << std::endl;
for (int i = 0; i < argc; i++)
{
std::cout << "\t" << i << ") \"" << argv[i] << "\"" << std::endl;
}
std::cerr << "Nothing is wrong! I just wanted to try writing to the std error stream." << std::endl;
return 100;
}
The application has been added to the solution as a console application. To create your own console application, please refer to the Console Application Sample.
To be able to find and invoke the SampleConsoleApplication exe at runtime, we need to package the output exe with the AppX. We've already added it to the sample application. However, to do the same in your own application, you can follow these steps:
- Put the output exe in the same folder as the C# sample. Right click on the C++ console app project, select Properties, then open the General tab
- Change the output directory to the be the same as the C# sample (or a subfolder) and click OK
- Build the console application to create and put the exe in its new location
- Add the exe to the UWP projct: Right click on the Project, select Add -> Existing Item
- Browse to and select the exe just built
- After the exe is added to the UWP project, we need to make sure it's added to the AppX package layout, Right click on the exe, select Properties
- Change Build Action to Content
- Change Copy to Output Directory to Always
- Finaly, to ensure the exe is built every time the main app is deployed, we need to change the project dependency, Right click the solution, select Project Dependencies and make the UWP project have a dependency on the console app
Now, everytime the solution is built or deployed, it'll ensure the console application exe is up-to-date and deployed with the rest of the AppX.
When a relative path is specified; such as SampleConsoleApp.exe, .\SampleConsoleApp.exe or a subfolder, subfolder\SampleConsoleApp.exe, the ProcessLauncher
API looks for the exe file starting at the package root of the AppX deployed.
However, the API is also capable of running apps anywhere on the device, by specifying the absolute path; e.g. c:\windows.system32\ipconfig.exe
To use an external exe, that exe must be added to an allow list. This is needed to prevent malicious apps from running.
To add an exe to the allow list, add to or replace the reg value in the reg command below:
reg.exe ADD "HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\EmbeddedMode\ProcessLauncher" /v AllowedExecutableFilesList /t REG_MULTI_SZ /d "c:\windows\system32\ipconfig.exe\0c:\windows\system32\tlist.exe\0"
Note, ProcessLauncher
API launches executables under the current user credentials, or DefautlAccount, so apps requiring admin priviliges will fail to run properly.
Go ahead, run the command above on your device, using SSHor PowerShell. Then, in the sample app (no need to restart) type the command c:\windows\system32\ipconfig.exe
. You should get an output similar to the one below:
Additionally, to overcome the limitations of using reg commands, the Process Launcher Whitelist Configuration tool can be used to very easily add or remove executables to the ProcessLauncher allow list.
The tool is available as part of the Windows 10 IoT Utilities on GitHub.
To use the tool:
- Clone or download a zip of the Windows 10 IoT Utilities from GitHub.
- Open and build the solution under ProcessLauncherWhitelist.
- Copy the executable to your device using FTP.
- Run it on the device using SSH or PowerShell.