GnsSharp.Gns.Posix32
0.1.0-alpha.10
dotnet add package GnsSharp.Gns.Posix32 --version 0.1.0-alpha.10
NuGet\Install-Package GnsSharp.Gns.Posix32 -Version 0.1.0-alpha.10
<PackageReference Include="GnsSharp.Gns.Posix32" Version="0.1.0-alpha.10" />
<PackageVersion Include="GnsSharp.Gns.Posix32" Version="0.1.0-alpha.10" />
<PackageReference Include="GnsSharp.Gns.Posix32" />
paket add GnsSharp.Gns.Posix32 --version 0.1.0-alpha.10
#r "nuget: GnsSharp.Gns.Posix32, 0.1.0-alpha.10"
#addin nuget:?package=GnsSharp.Gns.Posix32&version=0.1.0-alpha.10&prerelease
#tool nuget:?package=GnsSharp.Gns.Posix32&version=0.1.0-alpha.10&prerelease
GnsSharp
GnsSharp is a C# binding for the ValveSoftware/GameNetworkingSockets.
It supports both the stand-alone open source GameNetworkingSockets, and the Steamworks SDK version of it,
So you can switch between those without too much hassle.
This project is heavily WIP.
API might be missing, subject to change, and things might break or not work at all;
Use it at your own risk.
Why?
None of the existing C# bindings support switching between stand-alone GNS and Steamworks SDK version of it.
So, I'm creating one for my own needs.
State of binding
Currently, I've fully ported the APIs that are compatible for both the stand-alone GNS and Steamworks SDK.
Steamworks exclusive features are much huge in scope, and I've only ported few APIs which I need.
More APIs might be added in the future if I need another Steamworks exclusive features.
<details> <summary>Table of state</summary>
| Interfaces | Ported? | Interfaces | Ported? | |---------------------|---------|--------------------------|---------| | ISteamApps | ❌ | ISteamNetworkingMessages | ✔ | | ISteamAppTicket | ❌ | ISteamNetworkingSockets | ✔ | | ISteamClient | ❌ | ISteamNetworkingUtils | ✔ | | ISteamController | ❌ | ISteamRemotePlay | ❌ | | ISteamFriends | ✔ | ISteamRemoteStorage | ✔ | | ISteamGameCoordinator | ❌ | ISteamScreenshots | ❌ | | ISteamGameServer | ❌ | ISteamTimeline | ❌ | | ISteamGameServerStats | ❌ | ISteamUGC | ❌ | | ISteamHTMLSurface | ❌ | ISteamUser | ✔ | | ISteamHTTP | ❌ | ISteamUserStats | ✔ | | ISteamInput | ❌ | ISteamUtils | ✔ | | ISteamInventory | ❌ | ISteamVideo | ❌ | | ISteamMatchmaking | ✔ | SteamEncryptedAppTicket | ❌ | | ISteamMatchmakingServers | ❌ | steam_api | ✔ | | ISteamMusic | ❌ | steam_gameserver | ✔ | | ISteamMusicRemote | ❌ | GameNetworkingSockets | ✔ | </details>
Documentation
Most of the APIs are pretty much the same as the original GNS, so you can check out the official Steam Networking Docs and GameNetworkingSockets examples to learn how to use them.
For Steamworks exclusive features, you can also refer to the Steamworks API overview and Reference.
Integrate
There are 8 different nuget packages available for different backends & platforms.
dotnet add package GnsSharp.Gns.Win64 --prerelease # Open source GNS for Windows 64-bit
dotnet add package GnsSharp.Gns.Win32 --prerelease # Open source GNS for Windows 32-bit
dotnet add package GnsSharp.Gns.Posix64 --prerelease # Open source GNS for POSIX 64-bit
dotnet add package GnsSharp.Gns.Posix32 --prerelease # Open source GNS for POSIX 32-bit
dotnet add package GnsSharp.Steamworks.Win64 --prerelease # Steamworks SDK for Windows 64-bit
dotnet add package GnsSharp.Steamworks.Win32 --prerelease # Steamworks SDK for Windows 32-bit
dotnet add package GnsSharp.Steamworks.Posix64 --prerelease # Steamworks SDK for POSIX 64-bit
dotnet add package GnsSharp.Steamworks.Posix32 --prerelease # Steamworks SDK for POSIX 32-bit
This is because GNS uses different struct pack size for each platform, and size_t
is different between 64/32 bits.
GnsSharp exclusive notes
Current supported versions are GameNetworkingSockets commit 725e273
and Steamworks SDK v1.62
Where's the native libraries?
I don't provide them here, for many reasons:
- Stand-alone open source GNS can be built in many different setups.
- Your Linux distro might provide older OpenSSL, which wouldn't work with my build of GNS. Or vice versa.
- Or you might want to use BCrypt instead. Or libsodium.
- You might want to try out the build with P2P enabled (btw, it's broken for current master commit
725e273
).
- Your Linux distro might provide older OpenSSL, which wouldn't work with my build of GNS. Or vice versa.
- I don't have any device with macOS.
- You might want to use the more recent version of GNS.
So, bring your own native dll/dylib/so files by building the GameNetworkingSockets on your own, or downloading the Steamworks SDK from the Steamworks partner site.
Can you provide the pre-built native libraries?
You can download the pre-built native libraries in nalchi repository.
Download the one for your platform, that ends with -Shared
.
It will also include nalchi, my efficient message sending utility library, but you can ignore that if you don't need it.
If you want to try out nalchi too, you can use it with my C# binding called NalchiSharp.
Initialization
This method works with both the open source GNS & Steamworks SDK:
// Initialize GNS or SteamAPI
bool initialized = false;
string? errMsg = null;
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.OpenSource)
{
initialized = GameNetworkingSockets.Init(out errMsg);
}
else if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
{
// For test environment, write `480` in `steam_appid.txt`, and put it next to your executable.
// And you must be running Steam client on your PC.
//
// Alternatively, you can uncomment the two lines below.
// Environment.SetEnvironmentVariable("SteamAppId", "480");
// Environment.SetEnvironmentVariable("SteamGameId", "480");
initialized = (SteamAPI.InitEx(out errMsg) == ESteamAPIInitResult.OK);
}
if (!initialized)
{
Console.WriteLine(errMsg!);
throw new Exception(errMsg!);
}
You might want to add NativeLibrary.Load()
too, depending on your environment.
Check out the "Self connect example" below for a runnable example.
Deinitialization
Again, this method works with both the open source GNS & Steamworks SDK:
// De-initialize GNS or SteamAPI
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.OpenSource)
GameNetworkingSockets.Kill();
else if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
SteamAPI.Shutdown();
Check out the "Self connect example" below for a runnable example.
Steam Callbacks
Steam callbacks are implemented as C# events in the respective Steam interfaces.
// Subscribe an event to receive Steam callbacks associated with it.
ISteamNetworkingSockets.User!.SteamNetAuthenticationStatusChanged
+= (ref SteamNetAuthenticationStatus_t data) {
...
}
There's 3 exception to this:
* FnSteamNetConnectionStatusChanged
* FnPtrSteamNetworkingMessagesSessionRequest
* FnPtrSteamNetworkingMessagesSessionFailed
// This delegate should be alive until the listen socket is closed.
FnSteamNetConnectionStatusChanged listenSocketStatusChanged = /* your callback handler */;
// Setup this delegate object as a listen socket's configuration.
Span<SteamNetworkingConfigValue_t> listenSocketConfigs = stackalloc SteamNetworkingConfigValue_t[1];
listenSocketConfigs[0].SetPtr(ESteamNetworkingConfigValue.Callback_ConnectionStatusChanged,
Marshal.GetFunctionPointerForDelegate(listenSocketStatusChanged));
// Start listening with specifying this configuration.
HSteamListenSocket listener
= ISteamNetworkingSockets.User!.CreateListenSocketIP(in address, listenSocketConfigs);
These exceptions are for 2 reasons:
- To allow the listen socket and the client connections to have a different callback set up.
- Because there's no Steam Callbacks in open source GNS, and I wanted to guarantee the same API usage for both versions.
Steam CallResults
Steam CallResults are implemented with custom awaitable CallTask<T>
, so you can await
it.
// Request the Steam async API call, and get a task associated with it
CallTask<NumberOfCurrentPlayers_t>? callTask = ISteamUserStats.User!.GetNumberOfCurrentPlayers();
if (callTask != null)
{
// Await for the task to complete
NumberOfCurrentPlayers_t? result = await callTask;
// If succeeded, print the result
if (result.HasValue && result.Value.Success)
Console.WriteLine($"Number of current players: {result.Value.Players}");
}
<details> <summary>Read spacewar cloud file async example</summary>
async Task ReadSpacewarCloudFileAsync()
{
string fileName = "message.dat";
// Assuming properly initialized, and running callback seperately
var storage = ISteamRemoteStorage.User!;
// Get the file size first.
int size = storage.GetFileSize(fileName);
if (size == 0)
{
Console.WriteLine($"File '{fileName}' doesn't exist");
return;
}
Console.WriteLine($"Size of '{fileName}' was {size}");
// Start reading file asynchronously.
CallTask<RemoteStorageFileReadAsyncComplete_t>? readTask
= storage.FileReadAsync(fileName, 0, (uint)size);
// Skip if failed to start reading file.
if (readTask == null)
{
Console.WriteLine("File read not initiated");
return;
}
// Await for reading to complete.
RemoteStorageFileReadAsyncComplete_t? complete = await readTask;
// Skip if reading failed.
if (!complete.HasValue)
{
Console.WriteLine("File read not complete");
return;
}
if (complete.Value.Result != EResult.OK)
{
Console.WriteLine($"File read not complete: {complete.Value.Result}");
return;
}
// Allocate buffer to copy the read bytes.
Span<byte> raw = stackalloc byte[(int)complete.Value.ReadSize];
// Copy the result to this buffer.
if (storage.FileReadAsyncComplete(complete.Value.FileReadAsync, raw))
{
// Assuming it's a UTF-8 string, print it.
string str = Encoding.UTF8.GetString(raw);
Console.WriteLine($"Read string: {str}");
}
else
{
Console.WriteLine("FileReadAsyncComplete() failed");
}
}
</details>
Build
Stand-alone GameNetworkingSockets
- Build the open source GameNetworkingSockets.
- Currently, GnsSharp uses commit
725e273
- Refer to the
BUILDING.md
on GameNetworkingSockets for details.- If you're using Developer Powershell for VS 2022 on Windows, do note that it defaults to x86 environment, which obviously doesn't work when building for the AMD64.
You need to switch to AMD64 environment manually with following:Enter-VsDevShell -DevCmdArguments "-arch=x64 -host_arch=x64" -VsInstallPath "C:/Program Files/Microsoft Visual Studio/2022/Community" -SkipAutomaticLocation
- If you're using Developer Powershell for VS 2022 on Windows, do note that it defaults to x86 environment, which obviously doesn't work when building for the AMD64.
- Currently, GnsSharp uses commit
- Copy the native library files to your executable's build directory.
- On Windows, you should copy all the dependent dlls along with the
GameNetworkingSockets.dll
.
- On Windows, you should copy all the dependent dlls along with the
Steamworks SDK
- Download the Steamworks SDK from the Steamworks partner site.
- Currently, GnsSharp uses Steamworks SDK v1.62
- Copy the native library files from the
sdk/redistributable_bin/
to your executable's build directory.
Runnable Examples
<details> <summary>Self connect example</summary>
using GnsSharp;
using System.Runtime.InteropServices;
using System.Text;
#pragma warning disable CS0162 // Unreachable code (because of `GnsSharpCore.Backend` check)
// Load the native library depending on your platform (it might be unnecessary, but just in case)
string nativeLibraryPath = Path.Join(AppContext.BaseDirectory,
"GameNetworkingSockets.dll" /* Open source GNS for Windows */
// "libGameNetworkingSockets.so" /* Open source GNS for Linux */
// "libGameNetworkingSockets.dylib" /* Open source GNS for macOS */
// "steam_api64.dll" /* Steamworks SDK for Windows (AMD64) */
// "steam_api.dll" /* Steamworks SDK for Windows (x86) */
// "libsteam_api.so" /* Steamworks SDK for Linux */
// "libsteam_api.dylib" /* Steamworks SDK for macOS */
);
IntPtr nativeLibrary = NativeLibrary.Load(nativeLibraryPath);
// Initialize GNS or SteamAPI
bool initialized = false;
string? errMsg = null;
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.OpenSource)
{
initialized = GameNetworkingSockets.Init(out errMsg);
}
else if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
{
// For test environment, write `480` in `steam_appid.txt`, and put it next to your executable.
// And you must be running Steam client on your PC.
//
// Alternatively, you can uncomment the two lines below.
// Environment.SetEnvironmentVariable("SteamAppId", "480");
// Environment.SetEnvironmentVariable("SteamGameId", "480");
initialized = (SteamAPI.InitEx(out errMsg) == ESteamAPIInitResult.OK);
}
if (!initialized)
{
Console.WriteLine(errMsg!);
throw new Exception(errMsg!);
}
// Run callbacks as a seperate task
CancellationTokenSource cancelTokenSrc = new();
CancellationToken cancelToken = cancelTokenSrc.Token;
Task callbackRunner;
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.OpenSource)
{
callbackRunner = Task.Run(async () =>
{
while (!cancelToken.IsCancellationRequested)
{
ISteamNetworkingSockets.User!.RunCallbacks();
await Task.Delay(16, cancelToken);
}
}, cancelToken);
}
else if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
{
callbackRunner = Task.Run(async () =>
{
while (!cancelToken.IsCancellationRequested)
{
SteamAPI.RunCallbacks();
await Task.Delay(16, cancelToken);
}
}, cancelToken);
}
// Setup the debug output delegate
// (For every callback delegate, including this one,
// it should be stored somewhere safe to prevent it from garbage collected.)
FSteamNetworkingSocketsDebugOutput debugOutput = (ESteamNetworkingSocketsDebugOutputType level, string msg) =>
{
Console.WriteLine($"[{level}] {msg}");
};
ISteamNetworkingUtils.User!.SetDebugOutputFunction(ESteamNetworkingSocketsDebugOutputType.Everything, debugOutput);
// Setup listen address: IPv6 any address & port 43000
SteamNetworkingIPAddr addr = default;
addr.ParseString("[::]:43000");
int serverClosing = 0;
// Setup listen socket connection status changed callback
FnSteamNetConnectionStatusChanged listenStatusChanged = (ref SteamNetConnectionStatusChangedCallback_t status) =>
{
switch (status.Info.State)
{
case ESteamNetworkingConnectionState.Connecting:
ISteamNetworkingSockets.User!.AcceptConnection(status.Conn);
Console.WriteLine("Server has accepted the connection from client!");
break;
case ESteamNetworkingConnectionState.ClosedByPeer:
case ESteamNetworkingConnectionState.ProblemDetectedLocally:
StringBuilder builder = new();
builder.Append($"Server: #{status.Conn} disconnected");
if (status.Info.EndDebug != null)
builder.Append($": {status.Info.EndDebug}");
Console.WriteLine(builder.ToString());
// Server side also need to close the connection to clean up resources
ISteamNetworkingSockets.User!.CloseConnection(status.Conn, 0, "Server's closing too!", false);
Interlocked.Exchange(ref serverClosing, 1);
break;
}
};
Span<SteamNetworkingConfigValue_t> serverConfigs = stackalloc SteamNetworkingConfigValue_t[1];
serverConfigs[0].SetPtr(ESteamNetworkingConfigValue.Callback_ConnectionStatusChanged,
Marshal.GetFunctionPointerForDelegate(listenStatusChanged));
// Server: Start listening
HSteamListenSocket listener = ISteamNetworkingSockets.User!.CreateListenSocketIP(in addr, serverConfigs);
serverConfigs[0].Dispose(); // Dispose config after usage, actually not required unless `SetString()` is used
// On Steam, in order to properly listen, you need to wait for the authentication to complete
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
{
for (int i = 0; ; ++i)
{
ESteamNetworkingAvailability avail =
ISteamNetworkingSockets.User.GetAuthenticationStatus(out SteamNetAuthenticationStatus_t auth);
if (avail == ESteamNetworkingAvailability.Current)
break;
else if (avail == ESteamNetworkingAvailability.Failed)
{
Console.WriteLine($"Auth failed: {auth.DebugMsg}");
throw new Exception(auth.DebugMsg);
}
Console.WriteLine($"Waiting for Steam authentication for {i} seconds... ({avail})");
await Task.Delay(1000);
}
Console.WriteLine("Steam authentication succeeded!");
}
// Setup connect address: IPv6 loopback address & port 43000
addr.ParseString("[::1]:43000");
int clientConnected = 0;
// Setup connect client connection status changed callback
FnSteamNetConnectionStatusChanged clientStatusChanged = (ref SteamNetConnectionStatusChangedCallback_t status) =>
{
switch (status.Info.State)
{
case ESteamNetworkingConnectionState.Connected:
Console.WriteLine("Client successfully connected to the server!");
Interlocked.Exchange(ref clientConnected, 1);
break;
case ESteamNetworkingConnectionState.ClosedByPeer:
case ESteamNetworkingConnectionState.ProblemDetectedLocally:
StringBuilder builder = new();
builder.Append($"Client: #{status.Conn} disconnected");
if (status.Info.EndDebug != null)
builder.Append($": {status.Info.EndDebug}");
Console.WriteLine(builder.ToString());
ISteamNetworkingSockets.User.CloseConnection(status.Conn, 0, "Client closing lately?", false);
break;
}
};
Span<SteamNetworkingConfigValue_t> clientConfigs = stackalloc SteamNetworkingConfigValue_t[1];
clientConfigs[0].SetPtr(ESteamNetworkingConfigValue.Callback_ConnectionStatusChanged,
Marshal.GetFunctionPointerForDelegate(clientStatusChanged));
// Client: Connect to the server
HSteamNetConnection client = ISteamNetworkingSockets.User.ConnectByIPAddress(addr, clientConfigs);
clientConfigs[0].Dispose();
// Wait for the connection to complete
while (clientConnected == 0)
await Task.Delay(16);
// Close from the client side
ISteamNetworkingSockets.User.CloseConnection(client, 0, "Client's closing!", false);
// Wait for the server side to close the connection
while (serverClosing == 0)
await Task.Delay(16);
// Stop the callback loop task
cancelTokenSrc.Cancel();
try
{
await callbackRunner;
}
catch (TaskCanceledException)
{
Console.WriteLine("Callback loop task stopped!");
}
// De-initialize GNS or SteamAPI
if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.OpenSource)
GameNetworkingSockets.Kill();
else if (GnsSharpCore.Backend == GnsSharpCore.BackendKind.Steamworks)
SteamAPI.Shutdown();
// Free the native library
NativeLibrary.Free(nativeLibrary);
</details>
License
GnsSharp is licensed under the MIT License.
This project depends on either the stand-alone GameNetworkingSockets or the Steamworks SDK.
- Stand-alone GameNetworkingSockets is licensed under the BSD 3-Clause "New" or "Revised" License.
- Refer to the GameNetworkingSockets' GitHub repo for other dependencies' licenses.
- Steamworks SDK is licensed under the STEAMWORKS SDK license.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net9.0 is compatible. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net9.0
- No dependencies.
NuGet packages (1)
Showing the top 1 NuGet packages that depend on GnsSharp.Gns.Posix32:
Package | Downloads |
---|---|
NalchiSharp.Gns.Posix32
Utilities for efficient message sending over Valve's GameNetworkingSockets. |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
0.1.0-alpha.10 | 30 | 3/30/2025 |
0.1.0-alpha.9 | 85 | 3/28/2025 |
0.1.0-alpha.8 | 92 | 3/28/2025 |
0.1.0-alpha.7 | 420 | 3/26/2025 |
0.1.0-alpha.6 | 118 | 3/22/2025 |
0.1.0-alpha.5 | 62 | 3/21/2025 |
0.1.0-alpha.4 | 106 | 3/17/2025 |
0.1.0-alpha.3 | 97 | 3/16/2025 |
0.1.0-alpha.2 | 99 | 3/16/2025 |