domenica 5 aprile 2020

Set Up .NET Core 3.1 ARM Development Toolchain with Visual Studio Code

In the previous post, I described the steps followed to set up a remote development toolchain for c++ programming on an ARM-based microcontroller (in my case, a Friendly ELEC NanoPi M1 Plus, but the same steps should be valid on the more popular RaspberryPi ).



In this note, I'll try to do something similar but using the latest long time support version of dotnet core, 3.1 LTS (ver. 3.1.201 at the time of writing this post).

Pi Setup

As for the C++ setup, we will use ssh to connect our debugger to the Pi. To debug we'll need root privileges, and to obtain this result we need to enable ssh connection using root.
To get this result we need to edit the sshd_config configuration file by running:

sudo nano /etc/ssh/sshd_config

check if the line

PermitRootLogin yes

exists or, in the case, add it and then reboot the Pi

sudo reboot


.NET Core setup

First thing first we need to install .net core SDK on the Pi. Following the instruction available on the sdk's download page on our ssh command line we will run the following commands:

~$ sudo apt-get install curl libunwind8 gettext
        
~$ wget https://download.visualstudio.microsoft.com/download/pr/ccbcbf70-9911-40b1-a8cf-e018a13e720e/03c0621c6510f9c6f4cca6951f2cc1a4/dotnet-sdk-3.1.201-linux-arm.tar.gz                 
        

The next step is to extract the packages we downloaded, and make the dotnet commands available at the terminal

~$ mkdir -p $HOME/dotnet && tar zxf dotnet-sdk-3.1.201-linux-arm.tar.gz -C $HOME/dotnet
~$ export DOTNET_ROOT=$HOME/dotnet
~$ export PATH=$PATH:$HOME/dotnet
        

In order to verify the installation, in our ssh terminal, run the dotnet --info command. If everything is ok something like this should appear.

As stated in the .NET Core download page

The above commands will only make the .NET SDK commands available for the terminal session in which it was run.
You can edit your shell profile to permanently add the commands. There are a number of different shells available for Linux and each has a different profile. For example:
  • Bash Shell: ~/.bash_profile, ~/.bashrc
  • Korn Shell: ~/.kshrc or .profile
  • Z Shell: ~/.zshrc or .zprofile
Edit the appropriate source file for you shell and add :$HOME/dotnet to the end of the existing PATH statement. If no PATH statement is included, add a new line with export PATH=$PATH:$HOME/dotnet.
Also, add export DOTNET_ROOT=$HOME/dotnet to the end of the file.

VS Remote Debugger Setup

To install the vscode remote debugger on the NanoPi, on the ssh terminal rin the command:

~$ curl -sSL https://aka.ms/getvsdbgsh | /bin/sh /dev/stdin -v latest -l ~/vsdbg

Visual Studio Code remote debugging test

Now it's time to test the whole toolchain.

On our development machine (in my case an old MacBook Pro 15), we need the dotnet SDK installed, and, of course our favorite code editor Visual Studio Code.

To achieve the goal, we will use a simple console application named toolchain-test

In order to verify the machine on which we are running our simple test program we'll change the code as follows:

    using System;

    namespace toolchain_test
    {
        class Program
        {
            static void Main(string[] args)
            {
                var logName = Environment.GetEnvironmentVariable("LOGNAME");
                Console.WriteLine($"Hello World! from {logName}");
            }
        }
    }                
            

As we did for the C++ toolchain we will make use of rsync, a copying tool available on macOS, Linux, and now also on Windows 10 via the Windows Subsystem for Linux or WSL2 so our task.json and launch.json files will be:

        {
            "version": "2.0.0",
            "tasks": [
                {
                    "label": "build",
                    "command": "dotnet",
                    "type": "process",
                    "args": [
                        "build",
                        "${workspaceFolder}/toolchain-test/toolchain-test.csproj",
                        "/property:GenerateFullPaths=true",
                        "/consoleloggerparameters:NoSummary"
                    ],
                    "problemMatcher": "$msCompile"
                },  
                {
                    "label": "publish",
                    "command": "dotnet",
                    "dependsOn": "build",
                    "type": "process",
                    "args": [
                        "publish",
                        "-r",
                        "linux-arm",
                        "${workspaceFolder}/toolchain-test/toolchain-test.csproj",
                        "/property:GenerateFullPaths=true",
                        "/consoleloggerparameters:NoSummary"
                    ],
                    "problemMatcher": "$msCompile"
                },
                {
                    "label": "rsync",
                    "type": "shell",
                    "dependsOn": "publish",
                    "osx": {
                        "command": "rsync -r -a -v -e ssh --delete toolchain-test/bin/Debug/netcoreapp3.1/linux-arm/ pi@192.168.1.177:/home/pi/projects/toolchain-test/"
                    }
                },
                {
                    "label": "watch",
                    "command": "dotnet",
                    "type": "process",
                    "args": [
                        "watch",
                        "run",
                        "${workspaceFolder}/toolchain-test/toolchain-test.csproj",
                        "/property:GenerateFullPaths=true",
                        "/consoleloggerparameters:NoSummary"
                    ],
                    "problemMatcher": "$msCompile"
                }
            ]
        }        
    

        {
            // Use IntelliSense to find out which attributes exist for C# debugging
            // Use hover for the description of the existing attributes
            // For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md
            "version": "0.2.0",
            "configurations": [
                {
                    "name": ".NET Core Launch (console)",
                    "type": "coreclr",
                    "request": "launch",
                    "preLaunchTask": "build",
                    "program": "${workspaceFolder}/toolchain-test/bin/Debug/netcoreapp3.1/toolchain-test.dll",
                    "args": [],
                    "cwd": "${workspaceFolder}",
                    "console": "internalConsole",
                    "stopAtEntry": false
                },
                {
                    "name": ".NET Core Remote Launch - Standalone Application (console)",
                    "type": "coreclr",
                    "request": "launch",
                    "program": "toolchain-test",
                    "args": [],
                    "cwd": "~/projects/toolchain-test",
                    "stopAtEntry": false,
                    "console": "internalConsole",
                    "pipeTransport": {
                        "pipeCwd": "${workspaceRoot}",
                        "pipeProgram": "/usr/bin/ssh",
                        "pipeArgs": [
                            "-T",
                            "pi@192.168.1.177"
                        ],
                        "debuggerPath": "~/vsdbg/vsdbg"
                    },
                    "preLaunchTask": "rsync",
                },        
                {
                    "name": ".NET Core Attach",
                    "type": "coreclr",
                    "request": "attach",
                    "processId": "${command:pickProcess}"
                }
            ]
        }        
    

In the launch.json we have two configurations, a local and a remote one.

Selecting the local one:


and pressing F5, the program will stop at the breakpoint as we can see in the following picture:


and the program output will be:


Selecting the remote debug configuration, we will have instead:




As we can see, in the first test, the logName variable assumes the development machine LOGNAME environment variable value, while in the second it assumes the Pi's one.

The code of this note is available here on my Github repo.

As always

Enjoy,

and if we are still under COVID-19 attack, #staysafe, #StayAtHome.


Nessun commento:

Posta un commento