mirror of
https://github.com/S7NetPlus/s7netplus.git
synced 2026-02-17 22:38:27 +08:00
Compare commits
117 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ab6308eacd | ||
|
|
130eeadbd8 | ||
|
|
76a7ea04f7 | ||
|
|
4764c997ed | ||
|
|
cb24e9a046 | ||
|
|
10315b4b4c | ||
|
|
0774e124bf | ||
|
|
1ebffe08e7 | ||
|
|
f419df4d73 | ||
|
|
1969aac1b2 | ||
|
|
2f2dcf7281 | ||
|
|
07325db2fa | ||
|
|
eada47cd24 | ||
|
|
5e1ac8c7bf | ||
|
|
13544a1bcf | ||
|
|
6fc526b886 | ||
|
|
f227ad4b53 | ||
|
|
e4cc42fa51 | ||
|
|
689e7ffd96 | ||
|
|
8087b8d315 | ||
|
|
a55ceba679 | ||
|
|
eb1fad9333 | ||
|
|
0de9364dee | ||
|
|
9380ea85c3 | ||
|
|
22451bc440 | ||
|
|
e98ce005c5 | ||
|
|
11a40cc5e3 | ||
|
|
f79286b2d0 | ||
|
|
fadd7d0cb3 | ||
|
|
652ff3a9bb | ||
|
|
9c0fea721a | ||
|
|
2ec73224c1 | ||
|
|
a8ef47b475 | ||
|
|
55aa06a1fc | ||
|
|
7e631a713f | ||
|
|
0797c5858f | ||
|
|
f1ae0ea084 | ||
|
|
addf6068bb | ||
|
|
970e9d4395 | ||
|
|
c3934c3493 | ||
|
|
e5823f2806 | ||
|
|
97e27ccc2b | ||
|
|
9b1faa0123 | ||
|
|
54dadec75a | ||
|
|
8b8ad13464 | ||
|
|
714ac62ab1 | ||
|
|
088cd0a4a8 | ||
|
|
361db8be9d | ||
|
|
e26860b0c0 | ||
|
|
6e103cea63 | ||
|
|
c5023c10e4 | ||
|
|
b61ac32913 | ||
|
|
b27e1c9083 | ||
|
|
71f7f8b400 | ||
|
|
4aca9e4e53 | ||
|
|
0bb7c5351a | ||
|
|
c3f86c32a2 | ||
|
|
3d0dd693ba | ||
|
|
8ad25033d5 | ||
|
|
12e180ea2d | ||
|
|
5891a30c5d | ||
|
|
b3077b27e7 | ||
|
|
8126018afd | ||
|
|
4e4071f07f | ||
|
|
534d9fd69d | ||
|
|
8da292ad2f | ||
|
|
019aeb26dc | ||
|
|
670fb70b78 | ||
|
|
aa15145184 | ||
|
|
12ea402769 | ||
|
|
18402604d1 | ||
|
|
53f651a482 | ||
|
|
7558b9a691 | ||
|
|
3185d1fccf | ||
|
|
0d9ccea11b | ||
|
|
1fc6899905 | ||
|
|
18c3883dc0 | ||
|
|
1f26833244 | ||
|
|
7d212134e3 | ||
|
|
38b26e0ce1 | ||
|
|
cf94f8ad11 | ||
|
|
8becc562a8 | ||
|
|
296ead69c7 | ||
|
|
ebf3da6280 | ||
|
|
42194aa788 | ||
|
|
9c8b453326 | ||
|
|
49e4d3369a | ||
|
|
ee06bec0fb | ||
|
|
05ccb05f3a | ||
|
|
0d2817661e | ||
|
|
e869d19587 | ||
|
|
e7194bc470 | ||
|
|
5bc2c6c5e7 | ||
|
|
77dcb1778b | ||
|
|
14053e342a | ||
|
|
ab3bd87701 | ||
|
|
bc7c27e1d4 | ||
|
|
f0256fd0cb | ||
|
|
209148ab02 | ||
|
|
2fc9eaade3 | ||
|
|
ab70bfb041 | ||
|
|
e277cf6e6c | ||
|
|
43b29825a4 | ||
|
|
6aa0133081 | ||
|
|
868e719b78 | ||
|
|
3833d29c0e | ||
|
|
144f814794 | ||
|
|
82aaf7e2cb | ||
|
|
f47918946d | ||
|
|
142f1ba90e | ||
|
|
d99d0d0e6f | ||
|
|
ce9f9f9e08 | ||
|
|
ea1140314b | ||
|
|
5d3f01e59e | ||
|
|
9c3f95ce73 | ||
|
|
12281ec802 | ||
|
|
8df1a9c8cb |
134
.github/workflows/dotnet.yml
vendored
Normal file
134
.github/workflows/dotnet.yml
vendored
Normal file
@@ -0,0 +1,134 @@
|
||||
# yaml-language-server: $schema=https://json.schemastore.org/github-workflow.json
|
||||
|
||||
name: .NET
|
||||
|
||||
on:
|
||||
workflow_dispatch: # Allow running the workflow manually from the GitHub UI
|
||||
push:
|
||||
branches:
|
||||
- 'main' # Run the workflow when pushing to the main branch
|
||||
pull_request:
|
||||
branches:
|
||||
- '*' # Run the workflow for all pull requests
|
||||
release:
|
||||
types:
|
||||
- published # Run the workflow when a new GitHub release is published
|
||||
|
||||
env:
|
||||
DOTNET_SKIP_FIRST_TIME_EXPERIENCE: 1
|
||||
DOTNET_NOLOGO: true
|
||||
NuGetDirectory: ${{ github.workspace}}/nuget
|
||||
|
||||
defaults:
|
||||
run:
|
||||
shell: pwsh
|
||||
|
||||
jobs:
|
||||
create_nuget:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
with:
|
||||
fetch-depth: 0 # Get all history to allow automatic versioning
|
||||
|
||||
- name: Install GitVersion
|
||||
uses: gittools/actions/gitversion/setup@v0
|
||||
with:
|
||||
versionSpec: '6.x'
|
||||
includePrerelease: true
|
||||
preferLatestVersion: true
|
||||
|
||||
- name: Determine Version
|
||||
id: gitversion
|
||||
uses: gittools/actions/gitversion/execute@v0
|
||||
|
||||
- name: Setup .NET
|
||||
uses: actions/setup-dotnet@v3
|
||||
|
||||
- run: >
|
||||
dotnet pack
|
||||
--configuration Release
|
||||
/p:AssemblyVersion=${{ steps.gitversion.outputs.assemblySemVer }}
|
||||
/p:FileVersion=${{ steps.gitversion.outputs.assemblySemFileVer }}
|
||||
/p:InformationalVersion=${{ steps.gitversion.outputs.informationalVersion }}
|
||||
/p:PackageVersion=${{ steps.gitversion.outputs.semVer }}
|
||||
--output ${{ env.NuGetDirectory }}
|
||||
|
||||
- uses: actions/upload-artifact@v3
|
||||
with:
|
||||
name: nuget
|
||||
if-no-files-found: error
|
||||
retention-days: 7
|
||||
path: |
|
||||
${{ env.NuGetDirectory }}/*.nupkg
|
||||
${{ env.NuGetDirectory }}/*.snupkg
|
||||
|
||||
run_test:
|
||||
name: test-${{ matrix.os }}
|
||||
runs-on: ${{ matrix.os }}
|
||||
env:
|
||||
configuration: Release
|
||||
artifacts: ${{ github.workspace }}/artifacts
|
||||
strategy:
|
||||
matrix:
|
||||
os: [windows-latest, ubuntu-20.04, macos-latest]
|
||||
fail-fast: false
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
|
||||
- name: Install Snap7 Linux
|
||||
if: ${{ matrix.os == 'ubuntu-20.04' }}
|
||||
run: |
|
||||
sudo add-apt-repository ppa:gijzelaar/snap7
|
||||
sudo apt-get update
|
||||
sudo apt-get install libsnap7-1 libsnap7-dev
|
||||
|
||||
- name: Install Snap7 MacOs
|
||||
if: ${{ matrix.os == 'macos-latest' }}
|
||||
run: |
|
||||
brew install snap7
|
||||
|
||||
- name: Setup Dotnet
|
||||
uses: actions/setup-dotnet@v3
|
||||
with:
|
||||
dotnet-version: |
|
||||
6.x
|
||||
7.x
|
||||
|
||||
- name: Nuget Cache
|
||||
uses: actions/cache@v3
|
||||
with:
|
||||
path: ~/.nuget/packages
|
||||
# Look to see if there is a cache hit for the corresponding requirements file
|
||||
key: ${{ runner.os }}-nuget-${{ hashFiles('**/packages.lock.json') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-nuget
|
||||
|
||||
- name: Test
|
||||
run: dotnet test --nologo --verbosity normal --logger GitHubActions
|
||||
|
||||
deploy:
|
||||
# Publish only when creating a GitHub Release
|
||||
# https://docs.github.com/en/repositories/releasing-projects-on-github/managing-releases-in-a-repository
|
||||
# You can update this logic if you want to manage releases differently
|
||||
if: github.event_name == 'release'
|
||||
runs-on: ubuntu-latest
|
||||
needs: [ create_nuget, run_test ]
|
||||
steps:
|
||||
- uses: actions/download-artifact@v3
|
||||
with:
|
||||
name: nuget
|
||||
path: ${{ env.NuGetDirectory }}
|
||||
|
||||
- name: Setup .NET Core
|
||||
uses: actions/setup-dotnet@v3
|
||||
|
||||
# Publish all NuGet packages to NuGet.org
|
||||
# Use --skip-duplicate to prevent errors if a package with the same version already exists.
|
||||
# If you retry a failed workflow, already published packages will be skipped without error.
|
||||
- name: Publish NuGet package
|
||||
run: |
|
||||
foreach($file in (Get-ChildItem "${{ env.NuGetDirectory }}" -Recurse -Include *.nupkg)) {
|
||||
dotnet nuget push $file --api-key "${{ secrets.NUGET_APIKEY }}" --source https://api.nuget.org/v3/index.json --skip-duplicate
|
||||
}
|
||||
80
.github/workflows/test.yml
vendored
80
.github/workflows/test.yml
vendored
@@ -1,80 +0,0 @@
|
||||
name: Test
|
||||
|
||||
on:
|
||||
[pull_request, push]
|
||||
|
||||
jobs:
|
||||
|
||||
build_test:
|
||||
name: test-${{ matrix.os }}-${{ matrix.test-framework }}
|
||||
runs-on: ${{ matrix.os }}
|
||||
env:
|
||||
configuration: Release
|
||||
artifacts: ${{ github.workspace }}/artifacts
|
||||
DOTNET_NOLOGO : 1
|
||||
strategy:
|
||||
matrix:
|
||||
os: [windows-latest, ubuntu-latest, macos-latest]
|
||||
test-framework: [netcoreapp3.1, net5.0]
|
||||
include:
|
||||
- os: ubuntu-latest
|
||||
test-framework: netcoreapp3.1
|
||||
installSnap7: true
|
||||
dotnet-sdk: '3.1.x'
|
||||
- os: ubuntu-latest
|
||||
test-framework: net5.0
|
||||
installSnap7: true
|
||||
dotnet-sdk: '5.0.x'
|
||||
- os: macos-latest
|
||||
test-framework: netcoreapp3.1
|
||||
installSnap7: true
|
||||
dotnet-sdk: '3.1.x'
|
||||
- os: macos-latest
|
||||
test-framework: net5.0
|
||||
installSnap7: true
|
||||
dotnet-sdk: '5.0.x'
|
||||
- os: windows-latest
|
||||
test-framework: netcoreapp3.1
|
||||
dotnet-sdk: '3.1.x'
|
||||
- os: windows-latest
|
||||
test-framework: net5.0
|
||||
dotnet-sdk: '5.0.x'
|
||||
- os: windows-latest
|
||||
test-framework: net452
|
||||
dotnet-sdk: '5.0.x'
|
||||
fail-fast: false
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
|
||||
- name: Install Snap7 Linux
|
||||
if: ${{ matrix.installSnap7 && matrix.os == 'ubuntu-latest' }}
|
||||
run: |
|
||||
sudo add-apt-repository ppa:gijzelaar/snap7
|
||||
sudo apt-get update
|
||||
sudo apt-get install libsnap7-1 libsnap7-dev
|
||||
|
||||
- name: Install Snap7 MacOs
|
||||
if: ${{ matrix.installSnap7 && matrix.os == 'macos-latest' }}
|
||||
run: |
|
||||
brew install snap7
|
||||
|
||||
- name: Setup Dotnet
|
||||
uses: actions/setup-dotnet@v1
|
||||
with:
|
||||
dotnet-version: ${{ matrix.dotnet-sdk }}
|
||||
|
||||
- name: Nuget Cache
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ~/.nuget/packages
|
||||
# Look to see if there is a cache hit for the corresponding requirements file
|
||||
key: ${{ runner.os }}-${{ matrix.test-framework }}-nuget-${{ hashFiles('**/packages.lock.json') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-${{ matrix.test-framework }}-nuget
|
||||
|
||||
- name: Restore
|
||||
run: dotnet restore S7.Net.UnitTest
|
||||
|
||||
- name: Test
|
||||
run: dotnet test --no-restore --nologo --verbosity normal --logger GitHubActions --framework ${{ matrix.test-framework }}
|
||||
@@ -1,21 +0,0 @@
|
||||
assembly-informational-format: '{NuGetVersion}'
|
||||
mode: ContinuousDeployment
|
||||
branches:
|
||||
master:
|
||||
tag: rc
|
||||
increment: Minor
|
||||
feature:
|
||||
regex: features?[/-]
|
||||
tag: rc-{BranchName}
|
||||
increment: Minor
|
||||
pull-request:
|
||||
regex: (pull|pull\-requests|pr)[/-]
|
||||
tag: rc-pr-{BranchName}
|
||||
increment: Minor
|
||||
hotfix:
|
||||
regex: hotfix(es)?[/-]
|
||||
tag: rc
|
||||
increment: Patch
|
||||
develop:
|
||||
regex: dev(elop)?(ment)?$
|
||||
tag: b
|
||||
259
S7.Net.UnitTest/CommunicationTests/Clock.cs
Normal file
259
S7.Net.UnitTest/CommunicationTests/Clock.cs
Normal file
@@ -0,0 +1,259 @@
|
||||
using System;
|
||||
using System.Net;
|
||||
using System.Threading.Tasks;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using S7.Net.Protocol;
|
||||
|
||||
namespace S7.Net.UnitTest.CommunicationTests;
|
||||
|
||||
[TestClass]
|
||||
public class Clock
|
||||
{
|
||||
[TestMethod, Timeout(1000)]
|
||||
public async Task Read_Clock_Value()
|
||||
{
|
||||
var cs = new CommunicationSequence
|
||||
{
|
||||
ConnectionOpenTemplates.ConnectionRequestConfirm,
|
||||
ConnectionOpenTemplates.CommunicationSetup,
|
||||
{
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 1d
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 read clock
|
||||
// UserData header
|
||||
32 07 00 00 PDU1 PDU2
|
||||
// Parameter length
|
||||
00 08
|
||||
// Data length
|
||||
00 04
|
||||
|
||||
// Parameter
|
||||
// Head
|
||||
00 01 12
|
||||
// Length
|
||||
04
|
||||
// Method (Request/Response): Req
|
||||
11
|
||||
// Type request (4...) Function group timers (...7)
|
||||
47
|
||||
// Subfunction: read clock
|
||||
01
|
||||
// Sequence number
|
||||
00
|
||||
|
||||
// Data
|
||||
// Return code
|
||||
0a
|
||||
// Transport size
|
||||
00
|
||||
// Payload length
|
||||
00 00
|
||||
""",
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 2b
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 read clock response
|
||||
// UserData header
|
||||
32 07 00 00 PDU1 PDU2
|
||||
// Parameter length
|
||||
00 0c
|
||||
// Data length
|
||||
00 0e
|
||||
|
||||
// Parameter
|
||||
// Head
|
||||
00 01 12
|
||||
// Length
|
||||
08
|
||||
// Method (Request/Response): Res
|
||||
12
|
||||
// Type response (8...) Function group timers (...7)
|
||||
87
|
||||
// Subfunction: read clock
|
||||
01
|
||||
// Sequence number
|
||||
01
|
||||
// Data unit reference
|
||||
00
|
||||
// Last data unit? Yes
|
||||
00
|
||||
// Error code
|
||||
00 00
|
||||
|
||||
// Data
|
||||
// Error code
|
||||
ff
|
||||
// Transport size: OCTET STRING
|
||||
09
|
||||
// Length
|
||||
00 0a
|
||||
|
||||
// Timestamp
|
||||
// Reserved
|
||||
00
|
||||
// Year 1
|
||||
19
|
||||
// Year 2
|
||||
14
|
||||
// Month
|
||||
08
|
||||
// Day
|
||||
20
|
||||
// Hour
|
||||
11
|
||||
// Minute
|
||||
59
|
||||
// Seconds
|
||||
43
|
||||
// Milliseconds: 912..., Day of week: ...4
|
||||
91 24
|
||||
"""
|
||||
}
|
||||
};
|
||||
|
||||
static async Task Client(int port)
|
||||
{
|
||||
var conn = new Plc(IPAddress.Loopback.ToString(), port, new TsapPair(new Tsap(1, 2), new Tsap(3, 4)));
|
||||
await conn.OpenAsync();
|
||||
var time = await conn.ReadClockAsync();
|
||||
|
||||
Assert.AreEqual(new DateTime(2014, 8, 20, 11, 59, 43, 912), time);
|
||||
conn.Close();
|
||||
}
|
||||
|
||||
await Task.WhenAll(cs.Serve(out var port), Client(port));
|
||||
}
|
||||
|
||||
[TestMethod, Timeout(1000)]
|
||||
public async Task Write_Clock_Value()
|
||||
{
|
||||
var cs = new CommunicationSequence
|
||||
{
|
||||
ConnectionOpenTemplates.ConnectionRequestConfirm,
|
||||
ConnectionOpenTemplates.CommunicationSetup,
|
||||
{
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 27
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 read clock
|
||||
// UserData header
|
||||
32 07 00 00 PDU1 PDU2
|
||||
// Parameter length
|
||||
00 08
|
||||
// Data length
|
||||
00 0e
|
||||
|
||||
// Parameter
|
||||
// Head
|
||||
00 01 12
|
||||
// Length
|
||||
04
|
||||
// Method (Request/Response): Req
|
||||
11
|
||||
// Type request (4...) Function group timers (...7)
|
||||
47
|
||||
// Subfunction: write clock
|
||||
02
|
||||
// Sequence number
|
||||
00
|
||||
|
||||
// Data
|
||||
// Return code
|
||||
ff
|
||||
// Transport size
|
||||
09
|
||||
// Payload length
|
||||
00 0a
|
||||
|
||||
// Payload
|
||||
// Timestamp
|
||||
// Reserved
|
||||
00
|
||||
// Year 1
|
||||
19
|
||||
// Year 2
|
||||
14
|
||||
// Month
|
||||
08
|
||||
// Day
|
||||
20
|
||||
// Hour
|
||||
11
|
||||
// Minute
|
||||
59
|
||||
// Seconds
|
||||
43
|
||||
// Milliseconds: 912..., Day of week: ...4
|
||||
91 24
|
||||
""",
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 21
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 read clock response
|
||||
// UserData header
|
||||
32 07 00 00 PDU1 PDU2
|
||||
// Parameter length
|
||||
00 0c
|
||||
// Data length
|
||||
00 04
|
||||
|
||||
// Parameter
|
||||
// Head
|
||||
00 01 12
|
||||
// Length
|
||||
08
|
||||
// Method (Request/Response): Res
|
||||
12
|
||||
// Type response (8...) Function group timers (...7)
|
||||
87
|
||||
// Subfunction: write clock
|
||||
02
|
||||
// Sequence number
|
||||
01
|
||||
// Data unit reference
|
||||
00
|
||||
// Last data unit? Yes
|
||||
00
|
||||
// Error code
|
||||
00 00
|
||||
|
||||
// Data
|
||||
// Error code
|
||||
0a
|
||||
// Transport size: NONE
|
||||
00
|
||||
// Length
|
||||
00 00
|
||||
"""
|
||||
}
|
||||
};
|
||||
|
||||
static async Task Client(int port)
|
||||
{
|
||||
var conn = new Plc(IPAddress.Loopback.ToString(), port, new TsapPair(new Tsap(1, 2), new Tsap(3, 4)));
|
||||
await conn.OpenAsync();
|
||||
await conn.WriteClockAsync(new DateTime(2014, 08, 20, 11, 59, 43, 912));
|
||||
|
||||
conn.Close();
|
||||
}
|
||||
|
||||
await Task.WhenAll(cs.Serve(out var port), Client(port));
|
||||
}
|
||||
}
|
||||
28
S7.Net.UnitTest/CommunicationTests/ConnectionOpen.cs
Normal file
28
S7.Net.UnitTest/CommunicationTests/ConnectionOpen.cs
Normal file
@@ -0,0 +1,28 @@
|
||||
using System.Net;
|
||||
using System.Threading.Tasks;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using S7.Net.Protocol;
|
||||
|
||||
namespace S7.Net.UnitTest.CommunicationTests;
|
||||
|
||||
[TestClass]
|
||||
public class ConnectionOpen
|
||||
{
|
||||
[TestMethod]
|
||||
public async Task Does_Not_Throw()
|
||||
{
|
||||
var cs = new CommunicationSequence {
|
||||
ConnectionOpenTemplates.ConnectionRequestConfirm,
|
||||
ConnectionOpenTemplates.CommunicationSetup
|
||||
};
|
||||
|
||||
async Task Client(int port)
|
||||
{
|
||||
var conn = new Plc(IPAddress.Loopback.ToString(), port, new TsapPair(new Tsap(1, 2), new Tsap(3, 4)));
|
||||
await conn.OpenAsync();
|
||||
conn.Close();
|
||||
}
|
||||
|
||||
await Task.WhenAll(cs.Serve(out var port), Client(port));
|
||||
}
|
||||
}
|
||||
107
S7.Net.UnitTest/CommunicationTests/ConnectionOpenTemplates.cs
Normal file
107
S7.Net.UnitTest/CommunicationTests/ConnectionOpenTemplates.cs
Normal file
@@ -0,0 +1,107 @@
|
||||
namespace S7.Net.UnitTest.CommunicationTests;
|
||||
|
||||
internal static class ConnectionOpenTemplates
|
||||
{
|
||||
public static RequestResponsePair ConnectionRequestConfirm { get; } = new RequestResponsePair(
|
||||
"""
|
||||
// TPKT
|
||||
03 // Version
|
||||
00 // Reserved
|
||||
00 16 // Length
|
||||
|
||||
// CR
|
||||
11 // Number of bytes following
|
||||
E0 // CR / Credit
|
||||
00 00 // Destination reference, unused
|
||||
__ __ // Source reference, unused
|
||||
00 // Class / Option
|
||||
|
||||
// Source TSAP
|
||||
C1 // Parameter code
|
||||
02 // Parameter length
|
||||
TSAP_SRC_CHAN // Channel
|
||||
TSAP_SRC_POS // Position
|
||||
|
||||
// Destination TSAP
|
||||
C2 // Parameter code
|
||||
02 // Parameter length
|
||||
TSAP_DEST_CHAN // Channel
|
||||
TSAP_DEST_POS // Position
|
||||
|
||||
// PDU Size parameter
|
||||
C0 // Parameter code
|
||||
01 // Parameter length
|
||||
0A // 1024 byte PDU (2 ^ 10)
|
||||
""",
|
||||
"""
|
||||
// TPKT
|
||||
03 // Version
|
||||
00 // Reserved
|
||||
00 0B // Length
|
||||
|
||||
// CC
|
||||
06 // Length
|
||||
D0 // CC / Credit
|
||||
00 00 // Destination reference
|
||||
00 00 // Source reference
|
||||
00 // Class / Option
|
||||
"""
|
||||
);
|
||||
|
||||
public static RequestResponsePair CommunicationSetup { get; } = new RequestResponsePair(
|
||||
"""
|
||||
// TPKT
|
||||
03 // Version
|
||||
00 // Reserved
|
||||
00 19 // Length
|
||||
|
||||
// Data header
|
||||
02 // Length
|
||||
F0 // Data identifier
|
||||
80 // PDU number and end of transmission
|
||||
|
||||
// S7 header
|
||||
32 // Protocol ID
|
||||
01 // Message type job request
|
||||
00 00 // Reserved
|
||||
PDU1 PDU2 // PDU reference
|
||||
00 08 // Parameter length (Communication Setup)
|
||||
00 00 // Data length
|
||||
|
||||
// Communication Setup
|
||||
F0 // Function code
|
||||
00 // Reserved
|
||||
00 03 // Max AMQ caller
|
||||
00 03 // Max AMQ callee
|
||||
03 C0 // PDU size (960)
|
||||
""",
|
||||
"""
|
||||
// TPKT
|
||||
03 // Version
|
||||
00 // Reserved
|
||||
00 1B // Length
|
||||
|
||||
// Data header
|
||||
02 // Length
|
||||
F0 // Data identifier
|
||||
80 // PDU number and end of transmission
|
||||
|
||||
// S7 header
|
||||
32 // Protocol ID
|
||||
03 // Message type ack data
|
||||
00 00 // Reserved
|
||||
PDU1 PDU2 // PDU reference
|
||||
00 08 // Parameter length (Communication Setup)
|
||||
00 00 // Data length
|
||||
00 // Error class
|
||||
00 // Error code
|
||||
|
||||
// Communication Setup
|
||||
F0 // Function code
|
||||
00 // Reserved
|
||||
00 03 // Max AMQ caller
|
||||
00 03 // Max AMQ callee
|
||||
03 C0 // PDU size (960)
|
||||
"""
|
||||
);
|
||||
}
|
||||
57
S7.Net.UnitTest/CommunicationTests/ReadPlcStatus.cs
Normal file
57
S7.Net.UnitTest/CommunicationTests/ReadPlcStatus.cs
Normal file
@@ -0,0 +1,57 @@
|
||||
using System.Net;
|
||||
using System.Threading.Tasks;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using S7.Net.Protocol;
|
||||
|
||||
namespace S7.Net.UnitTest.CommunicationTests;
|
||||
|
||||
[TestClass]
|
||||
public class ReadPlcStatus
|
||||
{
|
||||
[TestMethod]
|
||||
public async Task Read_Status_Run()
|
||||
{
|
||||
var cs = new CommunicationSequence {
|
||||
ConnectionOpenTemplates.ConnectionRequestConfirm,
|
||||
ConnectionOpenTemplates.CommunicationSetup,
|
||||
{
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 21
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 SZL read
|
||||
32 07 00 00 PDU1 PDU2 00 08 00 08 00 01 12 04 11 44
|
||||
01 00 ff 09 00 04 04 24 00 00
|
||||
""",
|
||||
"""
|
||||
// TPKT
|
||||
03 00 00 3d
|
||||
|
||||
// COTP
|
||||
02 f0 80
|
||||
|
||||
// S7 SZL response
|
||||
32 07 00 00 PDU1 PDU2 00 0c 00 20 00 01 12 08 12 84
|
||||
01 02 00 00 00 00 ff 09 00 1c 04 24 00 00 00 14
|
||||
00 01 51 44 ff 08 00 00 00 00 00 00 00 00 14 08
|
||||
20 12 05 28 34 94
|
||||
"""
|
||||
}
|
||||
};
|
||||
|
||||
async Task Client(int port)
|
||||
{
|
||||
var conn = new Plc(IPAddress.Loopback.ToString(), port, new TsapPair(new Tsap(1, 2), new Tsap(3, 4)));
|
||||
await conn.OpenAsync();
|
||||
var status = await conn.ReadStatusAsync();
|
||||
|
||||
Assert.AreEqual(0x08, status);
|
||||
conn.Close();
|
||||
}
|
||||
|
||||
await Task.WhenAll(cs.Serve(out var port), Client(port));
|
||||
}
|
||||
}
|
||||
@@ -24,5 +24,21 @@ namespace S7.Net.UnitTest
|
||||
Assert.IsFalse(dummyByte.SelectBit(7));
|
||||
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void T01_TestSetBit()
|
||||
{
|
||||
byte dummyByte = 0xAA; // 1010 1010
|
||||
dummyByte.SetBit(0, true);
|
||||
dummyByte.SetBit(1, false);
|
||||
dummyByte.SetBit(2, true);
|
||||
dummyByte.SetBit(3, false);
|
||||
Assert.AreEqual<byte>(dummyByte, 0xA5);// 1010 0101
|
||||
dummyByte.SetBit(4, true);
|
||||
dummyByte.SetBit(5, true);
|
||||
dummyByte.SetBit(6, true);
|
||||
dummyByte.SetBit(7, true);
|
||||
Assert.AreEqual<byte>(dummyByte, 0xF5);// 1111 0101
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
7
S7.Net.UnitTest/Framework/IsExternalInit.cs
Normal file
7
S7.Net.UnitTest/Framework/IsExternalInit.cs
Normal file
@@ -0,0 +1,7 @@
|
||||
using System.ComponentModel;
|
||||
|
||||
namespace System.Runtime.CompilerServices
|
||||
{
|
||||
[EditorBrowsable(EditorBrowsableState.Never)]
|
||||
internal record IsExternalInit;
|
||||
}
|
||||
@@ -1,4 +1,6 @@
|
||||
|
||||
using S7.Net.Types;
|
||||
|
||||
namespace S7.Net.UnitTest.Helpers
|
||||
{
|
||||
class TestClass
|
||||
@@ -51,5 +53,16 @@ namespace S7.Net.UnitTest.Helpers
|
||||
/// DB1.DBD16
|
||||
/// </summary>
|
||||
public ushort DWordVariable { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// DB1.DBX20.0
|
||||
/// </summary>
|
||||
[S7String(S7StringType.S7WString, 10)]
|
||||
public string WStringVariable { get; set; }
|
||||
/// <summary>
|
||||
/// DB1.DBX44.0
|
||||
/// </summary>
|
||||
[S7String(S7StringType.S7String, 10)]
|
||||
public string StringVariable { get; set; }
|
||||
}
|
||||
}
|
||||
|
||||
82
S7.Net.UnitTest/Infrastructure/CommunicationSequence.cs
Normal file
82
S7.Net.UnitTest/Infrastructure/CommunicationSequence.cs
Normal file
@@ -0,0 +1,82 @@
|
||||
using System;
|
||||
using System.Buffers;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Net;
|
||||
using System.Net.Sockets;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace S7.Net.UnitTest;
|
||||
|
||||
internal class CommunicationSequence : IEnumerable<RequestResponsePair>
|
||||
{
|
||||
private readonly List<RequestResponsePair> _requestResponsePairs = new List<RequestResponsePair>();
|
||||
|
||||
public void Add(RequestResponsePair requestResponsePair)
|
||||
{
|
||||
_requestResponsePairs.Add(requestResponsePair);
|
||||
}
|
||||
|
||||
public void Add(string requestPattern, string responsePattern)
|
||||
{
|
||||
_requestResponsePairs.Add(new RequestResponsePair(requestPattern, responsePattern));
|
||||
}
|
||||
|
||||
public IEnumerator<RequestResponsePair> GetEnumerator()
|
||||
{
|
||||
return _requestResponsePairs.GetEnumerator();
|
||||
}
|
||||
|
||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
||||
|
||||
public Task Serve(out int port)
|
||||
{
|
||||
var socket = CreateBoundListenSocket(out port);
|
||||
socket.Listen(0);
|
||||
|
||||
async Task Impl()
|
||||
{
|
||||
await Task.Yield();
|
||||
var socketIn = socket.Accept();
|
||||
|
||||
var buffer = ArrayPool<byte>.Shared.Rent(1024);
|
||||
try
|
||||
{
|
||||
foreach (var pair in _requestResponsePairs)
|
||||
{
|
||||
var bytesReceived = socketIn.Receive(buffer, SocketFlags.None);
|
||||
|
||||
var received = buffer.Take(bytesReceived).ToArray();
|
||||
Console.WriteLine($"=> {BitConverter.ToString(received)}");
|
||||
|
||||
var response = Responder.Respond(pair, received);
|
||||
|
||||
Console.WriteLine($"<= {BitConverter.ToString(response)}");
|
||||
socketIn.Send(response);
|
||||
}
|
||||
}
|
||||
finally
|
||||
{
|
||||
ArrayPool<byte>.Shared.Return(buffer);
|
||||
}
|
||||
|
||||
socketIn.Close();
|
||||
}
|
||||
|
||||
return Impl();
|
||||
}
|
||||
|
||||
private static Socket CreateBoundListenSocket(out int port)
|
||||
{
|
||||
var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||
var endpoint = new IPEndPoint(IPAddress.Loopback, 0);
|
||||
|
||||
socket.Bind(endpoint);
|
||||
|
||||
var localEndpoint = (IPEndPoint)socket.LocalEndPoint!;
|
||||
port = localEndpoint.Port;
|
||||
|
||||
return socket;
|
||||
}
|
||||
}
|
||||
3
S7.Net.UnitTest/Infrastructure/RequestResponsePair.cs
Normal file
3
S7.Net.UnitTest/Infrastructure/RequestResponsePair.cs
Normal file
@@ -0,0 +1,3 @@
|
||||
namespace S7.Net.UnitTest;
|
||||
|
||||
internal record RequestResponsePair(string RequestPattern, string ResponsePattern);
|
||||
80
S7.Net.UnitTest/Infrastructure/Responder.cs
Normal file
80
S7.Net.UnitTest/Infrastructure/Responder.cs
Normal file
@@ -0,0 +1,80 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.IO;
|
||||
|
||||
namespace S7.Net.UnitTest;
|
||||
|
||||
internal static class Responder
|
||||
{
|
||||
private const string Comment = "//";
|
||||
private static char[] Space = " ".ToCharArray();
|
||||
|
||||
public static byte[] Respond(RequestResponsePair pair, byte[] request)
|
||||
{
|
||||
var offset = 0;
|
||||
var matches = new Dictionary<string, byte>();
|
||||
var res = new List<byte>();
|
||||
using var requestReader = new StringReader(pair.RequestPattern);
|
||||
|
||||
string line;
|
||||
while ((line = requestReader.ReadLine()) != null)
|
||||
{
|
||||
var tokens = line.Split(Space, StringSplitOptions.RemoveEmptyEntries);
|
||||
foreach (var token in tokens)
|
||||
{
|
||||
if (token.StartsWith(Comment)) break;
|
||||
|
||||
if (offset >= request.Length)
|
||||
{
|
||||
throw new Exception("Request pattern has more data than request.");
|
||||
}
|
||||
|
||||
var received = request[offset];
|
||||
|
||||
if (token.Length == 2 && byte.TryParse(token, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out var value))
|
||||
{
|
||||
// Number, exact match
|
||||
if (value != received)
|
||||
{
|
||||
throw new Exception($"Incorrect data at offset {offset}. Expected {value:X2}, received {received:X2}.");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
matches[token] = received;
|
||||
}
|
||||
|
||||
offset++;
|
||||
}
|
||||
}
|
||||
|
||||
if (offset != request.Length) throw new Exception("Request contained more data than request pattern.");
|
||||
|
||||
using var responseReader = new StringReader(pair.ResponsePattern);
|
||||
while ((line = responseReader.ReadLine()) != null)
|
||||
{
|
||||
var tokens = line.Split(Space, StringSplitOptions.RemoveEmptyEntries);
|
||||
foreach (var token in tokens)
|
||||
{
|
||||
if (token.StartsWith(Comment)) break;
|
||||
|
||||
if (token.Length == 2 && byte.TryParse(token, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out var value))
|
||||
{
|
||||
res.Add(value);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (!matches.TryGetValue(token, out var match))
|
||||
{
|
||||
throw new Exception($"Unmatched token '{token}' in response.");
|
||||
}
|
||||
|
||||
res.Add(match);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return res.ToArray();
|
||||
}
|
||||
}
|
||||
@@ -1,8 +1,14 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFrameworks>net452;netcoreapp3.1;net5.0</TargetFrameworks>
|
||||
<PropertyGroup Condition=" '$(OS)' != 'Windows_NT' ">
|
||||
<TargetFrameworks>net6.0;net7.0</TargetFrameworks>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(OS)' == 'Windows_NT' ">
|
||||
<TargetFrameworks>net6.0;net7.0;net462</TargetFrameworks>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<SignAssembly>true</SignAssembly>
|
||||
<AssemblyOriginatorKeyFile>Properties\S7.Net.snk</AssemblyOriginatorKeyFile>
|
||||
<IsPackable>false</IsPackable>
|
||||
@@ -11,7 +17,10 @@
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="GitHubActionsTestLogger" Version="1.1.0" />
|
||||
<PackageReference Include="GitHubActionsTestLogger" Version="2.3.2">
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.7.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="2.1.2" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="2.1.2" />
|
||||
|
||||
@@ -7,6 +7,12 @@ using S7.Net.Types;
|
||||
using S7.UnitTest.Helpers;
|
||||
using System.Threading.Tasks;
|
||||
using System.Threading;
|
||||
using System.Security.Cryptography;
|
||||
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
using System.Buffers;
|
||||
#endif
|
||||
|
||||
#endregion
|
||||
|
||||
@@ -138,6 +144,33 @@ namespace S7.Net.UnitTest
|
||||
CollectionAssert.AreEqual(data, readData);
|
||||
}
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
|
||||
/// <summary>
|
||||
/// Write/Read a large amount of data to test PDU max
|
||||
/// </summary>
|
||||
[TestMethod]
|
||||
public async Task Test_Async_WriteLargeByteArrayWithMemory()
|
||||
{
|
||||
Assert.IsTrue(plc.IsConnected, "Before executing this test, the plc must be connected. Check constructor.");
|
||||
|
||||
var randomEngine = new Random();
|
||||
using var dataOwner = MemoryPool<byte>.Shared.Rent(8192);
|
||||
var data = dataOwner.Memory.Slice(0, 8192);
|
||||
var db = 2;
|
||||
randomEngine.NextBytes(data.Span);
|
||||
|
||||
await plc.WriteBytesAsync(DataType.DataBlock, db, 0, data);
|
||||
|
||||
using var readDataOwner = MemoryPool<byte>.Shared.Rent(data.Length);
|
||||
var readData = readDataOwner.Memory.Slice(0, data.Length);
|
||||
await plc.ReadBytesAsync(readData, DataType.DataBlock, db, 0);
|
||||
|
||||
CollectionAssert.AreEqual(data.ToArray(), readData.ToArray());
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/// <summary>
|
||||
/// Read/Write a class that has the same properties of a DB with the same field in the same order
|
||||
/// </summary>
|
||||
@@ -154,7 +187,9 @@ namespace S7.Net.UnitTest
|
||||
IntVariable = -15000,
|
||||
LRealVariable = -154.789,
|
||||
RealVariable = -154.789f,
|
||||
DWordVariable = 850
|
||||
DWordVariable = 850,
|
||||
WStringVariable = "ÄÜÉÊéà",
|
||||
StringVariable = "Hallo"
|
||||
};
|
||||
|
||||
await plc.WriteClassAsync(tc, DB2);
|
||||
@@ -168,6 +203,8 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(tc.LRealVariable, tc2.LRealVariable);
|
||||
Assert.AreEqual(tc.RealVariable, tc2.RealVariable);
|
||||
Assert.AreEqual(tc.DWordVariable, tc2.DWordVariable);
|
||||
Assert.AreEqual(tc.WStringVariable, tc2.WStringVariable);
|
||||
Assert.AreEqual(tc.StringVariable, tc2.StringVariable);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
@@ -580,7 +617,9 @@ namespace S7.Net.UnitTest
|
||||
IntVariable = -15000,
|
||||
LRealVariable = -154.789,
|
||||
RealVariable = -154.789f,
|
||||
DWordVariable = 850
|
||||
DWordVariable = 850,
|
||||
WStringVariable = "ÄÜÉÊéà",
|
||||
StringVariable = "Hallo"
|
||||
};
|
||||
|
||||
await plc.WriteClassAsync(tc, DB2);
|
||||
@@ -628,7 +667,10 @@ namespace S7.Net.UnitTest
|
||||
IntVariable = -15000,
|
||||
LRealVariable = -154.789,
|
||||
RealVariable = -154.789f,
|
||||
DWordVariable = 850
|
||||
DWordVariable = 850,
|
||||
WStringVariable = "ÄÜÉÊéà",
|
||||
StringVariable = "Hallo"
|
||||
|
||||
};
|
||||
|
||||
await plc.WriteClassAsync(tc, DB2);
|
||||
@@ -646,6 +688,9 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(Math.Round(tc2.LRealVariable, 3), Math.Round(tc2Generic.LRealVariable, 3));
|
||||
Assert.AreEqual(tc2.RealVariable, tc2Generic.RealVariable);
|
||||
Assert.AreEqual(tc2.DWordVariable, tc2Generic.DWordVariable);
|
||||
Assert.AreEqual(tc2.WStringVariable, tc2Generic.WStringVariable);
|
||||
Assert.AreEqual(tc2.StringVariable, tc2Generic.StringVariable);
|
||||
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
@@ -671,7 +716,9 @@ namespace S7.Net.UnitTest
|
||||
IntVariable = -15000,
|
||||
LRealVariable = -154.789,
|
||||
RealVariable = -154.789f,
|
||||
DWordVariable = 850
|
||||
DWordVariable = 850,
|
||||
WStringVariable = "ÄÜÉÊéà",
|
||||
StringVariable = "Hallo"
|
||||
};
|
||||
|
||||
await plc.WriteClassAsync(tc, DB2);
|
||||
@@ -686,6 +733,8 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(Math.Round(tc2Generic.LRealVariable, 3), Math.Round(tc2GenericWithClassFactory.LRealVariable, 3));
|
||||
Assert.AreEqual(tc2Generic.RealVariable, tc2GenericWithClassFactory.RealVariable);
|
||||
Assert.AreEqual(tc2Generic.DWordVariable, tc2GenericWithClassFactory.DWordVariable);
|
||||
Assert.AreEqual(tc2Generic.WStringVariable, tc2GenericWithClassFactory.WStringVariable);
|
||||
Assert.AreEqual(tc2Generic.StringVariable, tc2GenericWithClassFactory.StringVariable);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
@@ -792,7 +841,9 @@ namespace S7.Net.UnitTest
|
||||
IntVariable = -15000,
|
||||
LRealVariable = -154.789,
|
||||
RealVariable = -154.789f,
|
||||
DWordVariable = 850
|
||||
DWordVariable = 850,
|
||||
WStringVariable = "ÄÜÉÊéà",
|
||||
StringVariable = "Hallo"
|
||||
};
|
||||
plc.WriteClass(tc, DB2);
|
||||
|
||||
@@ -914,6 +965,31 @@ namespace S7.Net.UnitTest
|
||||
}
|
||||
}
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
|
||||
[TestMethod]
|
||||
public async Task Test_Async_ReadWriteBytesManyWithMemory()
|
||||
{
|
||||
Assert.IsTrue(plc.IsConnected, "Before executing this test, the plc must be connected. Check constructor.");
|
||||
|
||||
using var data = MemoryPool<byte>.Shared.Rent(2000);
|
||||
for (int i = 0; i < data.Memory.Length; i++)
|
||||
data.Memory.Span[i] = (byte)(i % 256);
|
||||
|
||||
await plc.WriteBytesAsync(DataType.DataBlock, 2, 0, data.Memory);
|
||||
|
||||
using var readData = MemoryPool<byte>.Shared.Rent(data.Memory.Length);
|
||||
|
||||
await plc.ReadBytesAsync(readData.Memory.Slice(0, data.Memory.Length), DataType.DataBlock, 2, 0);
|
||||
|
||||
for (int x = 0; x < data.Memory.Length; x++)
|
||||
{
|
||||
Assert.AreEqual(x % 256, readData.Memory.Span[x], string.Format("Bit {0} failed", x));
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/// <summary>
|
||||
/// Write a large amount of data and test cancellation
|
||||
/// </summary>
|
||||
@@ -930,7 +1006,7 @@ namespace S7.Net.UnitTest
|
||||
var db = 2;
|
||||
randomEngine.NextBytes(data);
|
||||
|
||||
cancellationSource.CancelAfter(TimeSpan.FromMilliseconds(5));
|
||||
cancellationSource.CancelAfter(System.TimeSpan.FromMilliseconds(5));
|
||||
try
|
||||
{
|
||||
await plc.WriteBytesAsync(DataType.DataBlock, db, 0, data, cancellationToken);
|
||||
@@ -950,6 +1026,47 @@ namespace S7.Net.UnitTest
|
||||
Console.WriteLine("Task was not cancelled as expected.");
|
||||
}
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
|
||||
/// <summary>
|
||||
/// Write a large amount of data and test cancellation
|
||||
/// </summary>
|
||||
[TestMethod]
|
||||
public async Task Test_Async_WriteLargeByteArrayWithCancellationWithMemory()
|
||||
{
|
||||
Assert.IsTrue(plc.IsConnected, "Before executing this test, the plc must be connected. Check constructor.");
|
||||
|
||||
var cancellationSource = new CancellationTokenSource();
|
||||
var cancellationToken = cancellationSource.Token;
|
||||
|
||||
using var dataOwner = MemoryPool<byte>.Shared.Rent(8192);
|
||||
var data = dataOwner.Memory.Slice(0, 8192);
|
||||
var randomEngine = new Random();
|
||||
var db = 2;
|
||||
randomEngine.NextBytes(data.Span);
|
||||
|
||||
cancellationSource.CancelAfter(System.TimeSpan.FromMilliseconds(5));
|
||||
try
|
||||
{
|
||||
await plc.WriteBytesAsync(DataType.DataBlock, db, 0, data, cancellationToken);
|
||||
}
|
||||
catch (OperationCanceledException)
|
||||
{
|
||||
// everything is good, that is the exception we expect
|
||||
Console.WriteLine("Operation was cancelled as expected.");
|
||||
return;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
Assert.Fail($"Wrong exception type received. Expected {typeof(OperationCanceledException)}, received {e.GetType()}.");
|
||||
}
|
||||
|
||||
// Depending on how tests run, this can also just succeed without getting cancelled at all. Do nothing in this case.
|
||||
Console.WriteLine("Task was not cancelled as expected.");
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/// <summary>
|
||||
/// Write a large amount of data and test cancellation
|
||||
/// </summary>
|
||||
@@ -982,6 +1099,7 @@ namespace S7.Net.UnitTest
|
||||
};
|
||||
await plc.ReadMultipleVarsAsync(dataItems, CancellationToken.None);
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,6 +5,11 @@ using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using S7.Net.UnitTest.Helpers;
|
||||
using S7.Net.Types;
|
||||
using S7.UnitTest.Helpers;
|
||||
using System.Security.Cryptography;
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
using System.Buffers;
|
||||
#endif
|
||||
|
||||
#endregion
|
||||
|
||||
@@ -183,6 +188,9 @@ namespace S7.Net.UnitTest
|
||||
tc.LRealVariable = -154.789;
|
||||
tc.RealVariable = -154.789f;
|
||||
tc.DWordVariable = 850;
|
||||
tc.WStringVariable = "ÄÜÉÊéà";
|
||||
tc.StringVariable = "Hallo";
|
||||
|
||||
plc.WriteClass(tc, DB2);
|
||||
TestClass tc2 = new TestClass();
|
||||
// Values that are read from a class are stored inside the class itself, that is passed by reference
|
||||
@@ -194,6 +202,8 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(tc.LRealVariable, tc2.LRealVariable);
|
||||
Assert.AreEqual(tc.RealVariable, tc2.RealVariable);
|
||||
Assert.AreEqual(tc.DWordVariable, tc2.DWordVariable);
|
||||
Assert.AreEqual(tc.WStringVariable, tc2.WStringVariable);
|
||||
Assert.AreEqual(tc.StringVariable, tc2.StringVariable);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -577,6 +587,8 @@ namespace S7.Net.UnitTest
|
||||
tc.LRealVariable = -154.789;
|
||||
tc.RealVariable = -154.789f;
|
||||
tc.DWordVariable = 850;
|
||||
tc.WStringVariable = "ÄÜÉÊéà";
|
||||
tc.StringVariable = "Hallo";
|
||||
|
||||
plc.WriteClass(tc, DB2);
|
||||
|
||||
@@ -622,6 +634,8 @@ namespace S7.Net.UnitTest
|
||||
tc.LRealVariable = -154.789;
|
||||
tc.RealVariable = -154.789f;
|
||||
tc.DWordVariable = 850;
|
||||
tc.WStringVariable = "ÄÜÉÊéà";
|
||||
tc.StringVariable = "Hallo";
|
||||
|
||||
plc.WriteClass(tc, DB2);
|
||||
|
||||
@@ -637,6 +651,8 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(Math.Round(tc2.LRealVariable, 3), Math.Round(tc2Generic.LRealVariable, 3));
|
||||
Assert.AreEqual(tc2.RealVariable, tc2Generic.RealVariable);
|
||||
Assert.AreEqual(tc2.DWordVariable, tc2Generic.DWordVariable);
|
||||
Assert.AreEqual(tc2.WStringVariable, tc2Generic.WStringVariable);
|
||||
Assert.AreEqual(tc2.StringVariable, tc2Generic.StringVariable);
|
||||
}
|
||||
|
||||
[TestMethod, ExpectedException(typeof(PlcException))]
|
||||
@@ -665,6 +681,8 @@ namespace S7.Net.UnitTest
|
||||
tc.LRealVariable = -154.789;
|
||||
tc.RealVariable = -154.789f;
|
||||
tc.DWordVariable = 850;
|
||||
tc.WStringVariable = "ÄÜÉÊéà";
|
||||
tc.StringVariable = "Hallo";
|
||||
|
||||
plc.WriteClass(tc, DB2);
|
||||
|
||||
@@ -679,6 +697,8 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(Math.Round(tc2Generic.LRealVariable, 3), Math.Round(tc2GenericWithClassFactory.LRealVariable, 3));
|
||||
Assert.AreEqual(tc2Generic.RealVariable, tc2GenericWithClassFactory.RealVariable);
|
||||
Assert.AreEqual(tc2Generic.DWordVariable, tc2GenericWithClassFactory.DWordVariable);
|
||||
Assert.AreEqual(tc2Generic.WStringVariable, tc2GenericWithClassFactory.WStringVariable);
|
||||
Assert.AreEqual(tc2Generic.StringVariable, tc2GenericWithClassFactory.StringVariable);
|
||||
}
|
||||
|
||||
[TestMethod, ExpectedException(typeof(PlcException))]
|
||||
@@ -762,6 +782,33 @@ namespace S7.Net.UnitTest
|
||||
CollectionAssert.AreEqual(data, readData);
|
||||
}
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
|
||||
/// <summary>
|
||||
/// Write/Read a large amount of data to test PDU max
|
||||
/// </summary>
|
||||
[TestMethod]
|
||||
public void T33_WriteLargeByteArrayWithSpan()
|
||||
{
|
||||
Assert.IsTrue(plc.IsConnected, "Before executing this test, the plc must be connected. Check constructor.");
|
||||
|
||||
var randomEngine = new Random();
|
||||
using var dataOwner = MemoryPool<byte>.Shared.Rent(8192);
|
||||
var data = dataOwner.Memory.Span.Slice(0, 8192);
|
||||
var db = 2;
|
||||
randomEngine.NextBytes(data);
|
||||
|
||||
plc.WriteBytes(DataType.DataBlock, db, 0, data);
|
||||
|
||||
using var readDataOwner = MemoryPool<byte>.Shared.Rent(data.Length);
|
||||
var readData = readDataOwner.Memory.Span.Slice(0, data.Length);
|
||||
plc.ReadBytes(readData, DataType.DataBlock, db, 0);
|
||||
|
||||
CollectionAssert.AreEqual(data.ToArray(), readData.ToArray());
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
[TestMethod, ExpectedException(typeof(PlcException))]
|
||||
public void T18_ReadStructThrowsIfPlcIsNotConnected()
|
||||
{
|
||||
@@ -837,6 +884,9 @@ namespace S7.Net.UnitTest
|
||||
tc.LRealVariable = -154.789;
|
||||
tc.RealVariable = -154.789f;
|
||||
tc.DWordVariable = 850;
|
||||
tc.WStringVariable = "ÄÜÉÊéà";
|
||||
tc.StringVariable = "Hallo";
|
||||
|
||||
plc.WriteClass(tc, DB2);
|
||||
|
||||
int expectedReadBytes = (int)Types.Class.GetClassSize(tc);
|
||||
@@ -987,6 +1037,32 @@ namespace S7.Net.UnitTest
|
||||
}
|
||||
}
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
|
||||
[TestMethod]
|
||||
public void T27_ReadWriteBytesManyWithSpan()
|
||||
{
|
||||
Assert.IsTrue(plc.IsConnected, "Before executing this test, the plc must be connected. Check constructor.");
|
||||
|
||||
using var dataOwner = MemoryPool<byte>.Shared.Rent(2000);
|
||||
var data = dataOwner.Memory.Span;
|
||||
for (int i = 0; i < data.Length; i++)
|
||||
data[i] = (byte)(i % 256);
|
||||
|
||||
plc.WriteBytes(DataType.DataBlock, 2, 0, data);
|
||||
|
||||
using var readDataOwner = MemoryPool<byte>.Shared.Rent(data.Length);
|
||||
var readData = readDataOwner.Memory.Span.Slice(0, data.Length);
|
||||
plc.ReadBytes(readData, DataType.DataBlock, 2, 0);
|
||||
|
||||
for (int x = 0; x < data.Length; x++)
|
||||
{
|
||||
Assert.AreEqual(x % 256, readData[x], $"Mismatch at offset {x}, expected {x % 256}, actual {readData[x]}.");
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
[TestMethod]
|
||||
public void T28_ReadClass_DoesntCrash_When_ReadingLessThan1Byte()
|
||||
{
|
||||
@@ -1041,7 +1117,7 @@ namespace S7.Net.UnitTest
|
||||
Assert.AreEqual(test_value, test_value2, "Compare DateTimeLong Write/Read");
|
||||
}
|
||||
|
||||
#endregion
|
||||
#endregion
|
||||
|
||||
#region Private methods
|
||||
|
||||
|
||||
@@ -17,6 +17,19 @@ namespace S7.Net.UnitTest.TypeTests
|
||||
Assert.AreEqual(Class.GetClassSize(new TestClassUnevenSize(3, 17)), 10);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Ensure Uint32 is correctly parsed through ReadClass functions. Adresses issue https://github.com/S7NetPlus/s7netplus/issues/414
|
||||
/// </summary>
|
||||
[TestMethod]
|
||||
public void TestUint32Read()
|
||||
{
|
||||
var result = new TestUint32();
|
||||
var data = new byte[4] { 0, 0, 0, 5 };
|
||||
var bytesRead = Class.FromBytes(result, data);
|
||||
Assert.AreEqual(bytesRead, data.Length);
|
||||
Assert.AreEqual(5u, result.Value1);
|
||||
}
|
||||
|
||||
private class TestClassUnevenSize
|
||||
{
|
||||
public bool Bool { get; set; }
|
||||
@@ -29,5 +42,10 @@ namespace S7.Net.UnitTest.TypeTests
|
||||
Bools = new bool[bitCount];
|
||||
}
|
||||
}
|
||||
|
||||
private class TestUint32
|
||||
{
|
||||
public uint Value1 { get; set; }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
82
S7.Net.UnitTest/TypeTests/TimeSpanTests.cs
Normal file
82
S7.Net.UnitTest/TypeTests/TimeSpanTests.cs
Normal file
@@ -0,0 +1,82 @@
|
||||
using System;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
|
||||
namespace S7.Net.UnitTest.TypeTests
|
||||
{
|
||||
public static class TimeSpanTests
|
||||
{
|
||||
private static readonly TimeSpan SampleTimeSpan = new TimeSpan(12, 0, 59, 37, 856);
|
||||
|
||||
private static readonly byte[] SampleByteArray = { 0x3E, 0x02, 0xE8, 0x00 };
|
||||
|
||||
private static readonly byte[] SpecMinByteArray = { 0x80, 0x00, 0x00, 0x00 };
|
||||
|
||||
private static readonly byte[] SpecMaxByteArray = { 0x7F, 0xFF, 0xFF, 0xFF };
|
||||
|
||||
[TestClass]
|
||||
public class FromByteArray
|
||||
{
|
||||
[TestMethod]
|
||||
public void Sample()
|
||||
{
|
||||
AssertFromByteArrayEquals(SampleTimeSpan, SampleByteArray);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void SpecMinimum()
|
||||
{
|
||||
AssertFromByteArrayEquals(Types.TimeSpan.SpecMinimumTimeSpan, SpecMinByteArray);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void SpecMaximum()
|
||||
{
|
||||
AssertFromByteArrayEquals(Types.TimeSpan.SpecMaximumTimeSpan, SpecMaxByteArray);
|
||||
}
|
||||
|
||||
private static void AssertFromByteArrayEquals(TimeSpan expected, params byte[] bytes)
|
||||
{
|
||||
Assert.AreEqual(expected, Types.TimeSpan.FromByteArray(bytes));
|
||||
}
|
||||
}
|
||||
|
||||
[TestClass]
|
||||
public class ToByteArray
|
||||
{
|
||||
[TestMethod]
|
||||
public void Sample()
|
||||
{
|
||||
AssertToByteArrayEquals(SampleTimeSpan, SampleByteArray);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void SpecMinimum()
|
||||
{
|
||||
AssertToByteArrayEquals(Types.TimeSpan.SpecMinimumTimeSpan, SpecMinByteArray);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void SpecMaximum()
|
||||
{
|
||||
AssertToByteArrayEquals(Types.TimeSpan.SpecMaximumTimeSpan, SpecMaxByteArray);
|
||||
}
|
||||
|
||||
[TestMethod, ExpectedException(typeof(ArgumentOutOfRangeException))]
|
||||
public void ThrowsOnTimeBeforeSpecMinimum()
|
||||
{
|
||||
Types.TimeSpan.ToByteArray(TimeSpan.FromDays(-25));
|
||||
}
|
||||
|
||||
[TestMethod, ExpectedException(typeof(ArgumentOutOfRangeException))]
|
||||
public void ThrowsOnTimeAfterSpecMaximum()
|
||||
{
|
||||
Types.TimeSpan.ToByteArray(new TimeSpan(30, 15, 15, 15, 15));
|
||||
}
|
||||
|
||||
private static void AssertToByteArrayEquals(TimeSpan value, params byte[] expected)
|
||||
{
|
||||
CollectionAssert.AreEqual(expected, Types.TimeSpan.ToByteArray(value));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -55,6 +55,7 @@ namespace S7.Net
|
||||
/// See: https://tools.ietf.org/html/rfc905
|
||||
/// </summary>
|
||||
/// <param name="stream">The socket to read from</param>
|
||||
/// <param name="cancellationToken">A cancellation token that can be used to cancel the asynchronous operation.</param>
|
||||
/// <returns>COTP DPDU instance</returns>
|
||||
public static async Task<TPDU> ReadAsync(Stream stream, CancellationToken cancellationToken)
|
||||
{
|
||||
@@ -89,6 +90,7 @@ namespace S7.Net
|
||||
/// See: https://tools.ietf.org/html/rfc905
|
||||
/// </summary>
|
||||
/// <param name="stream">The stream to read from</param>
|
||||
/// <param name="cancellationToken">A cancellation token that can be used to cancel the asynchronous operation.</param>
|
||||
/// <returns>Data in TSDU</returns>
|
||||
public static async Task<byte[]> ReadAsync(Stream stream, CancellationToken cancellationToken)
|
||||
{
|
||||
|
||||
@@ -138,19 +138,59 @@ namespace S7.Net
|
||||
|
||||
/// <summary>
|
||||
/// Helper to get a bit value given a byte and the bit index.
|
||||
/// Example: DB1.DBX0.5 -> var bytes = ReadBytes(DB1.DBW0); bool bit = bytes[0].SelectBit(5);
|
||||
/// <br/>
|
||||
/// <example>
|
||||
/// Get the bit at DB1.DBX0.5:
|
||||
/// <code>
|
||||
/// byte data = ReadByte("DB1.DBB0");
|
||||
/// bool bit = data.SelectBit(5);
|
||||
/// </code>
|
||||
/// </example>
|
||||
/// </summary>
|
||||
/// <param name="data"></param>
|
||||
/// <param name="bitPosition"></param>
|
||||
/// <returns></returns>
|
||||
public static bool SelectBit(this byte data, int bitPosition)
|
||||
/// <param name="data">The data to get from.</param>
|
||||
/// <param name="index">The zero-based index of the bit to get.</param>
|
||||
/// <returns>The Boolean value will get.</returns>
|
||||
public static bool SelectBit(this byte data, int index)
|
||||
{
|
||||
int mask = 1 << bitPosition;
|
||||
int mask = 1 << index;
|
||||
int result = data & mask;
|
||||
|
||||
return (result != 0);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Helper to set a bit value to the given byte at the bit index.
|
||||
/// <br/>
|
||||
/// <example>
|
||||
/// Set the bit at index 4:
|
||||
/// <code>
|
||||
/// byte data = 0;
|
||||
/// data.SetBit(4, true);
|
||||
/// </code>
|
||||
/// </example>
|
||||
/// </summary>
|
||||
/// <param name="data">The data to be modified.</param>
|
||||
/// <param name="index">The zero-based index of the bit to set.</param>
|
||||
/// <param name="value">The Boolean value to assign to the bit.</param>
|
||||
public static void SetBit(this ref byte data, int index, bool value)
|
||||
{
|
||||
if ((uint)index > 7)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
if (value)
|
||||
{
|
||||
byte mask = (byte)(1 << index);
|
||||
data |= mask;
|
||||
}
|
||||
else
|
||||
{
|
||||
byte mask = (byte)~(1 << index);
|
||||
data &= mask;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts from ushort value to short value; it's used to retrieve negative values from words
|
||||
/// </summary>
|
||||
|
||||
@@ -202,10 +202,20 @@
|
||||
/// DateTIme variable type
|
||||
/// </summary>
|
||||
DateTime,
|
||||
|
||||
/// <summary>
|
||||
/// IEC date (legacy) variable type
|
||||
/// </summary>
|
||||
Date,
|
||||
|
||||
/// <summary>
|
||||
/// DateTimeLong variable type
|
||||
/// </summary>
|
||||
DateTimeLong
|
||||
DateTimeLong,
|
||||
|
||||
/// <summary>
|
||||
/// S7 TIME variable type - serialized as S7 DInt and deserialized as C# TimeSpan
|
||||
/// </summary>
|
||||
Time
|
||||
}
|
||||
}
|
||||
|
||||
23
S7.Net/Helper/DateTimeExtensions.cs
Normal file
23
S7.Net/Helper/DateTimeExtensions.cs
Normal file
@@ -0,0 +1,23 @@
|
||||
using System;
|
||||
using S7.Net.Types;
|
||||
using DateTime = System.DateTime;
|
||||
|
||||
namespace S7.Net.Helper
|
||||
{
|
||||
public static class DateTimeExtensions
|
||||
{
|
||||
public static ushort GetDaysSinceIecDateStart(this DateTime dateTime)
|
||||
{
|
||||
if (dateTime < Date.IecMinDate)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException($"DateTime must be at least {Date.IecMinDate:d}");
|
||||
}
|
||||
if (dateTime > Date.IecMaxDate)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException($"DateTime must be lower than {Date.IecMaxDate:d}");
|
||||
}
|
||||
|
||||
return (ushort)(dateTime - Date.IecMinDate).TotalDays;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,10 @@
|
||||
|
||||
using System;
|
||||
using System.Buffers;
|
||||
using System.IO;
|
||||
|
||||
namespace S7.Net.Helper
|
||||
{
|
||||
#if !NET5_0_OR_GREATER
|
||||
internal static class MemoryStreamExtension
|
||||
{
|
||||
/// <summary>
|
||||
@@ -10,9 +14,25 @@ namespace S7.Net.Helper
|
||||
/// </summary>
|
||||
/// <param name="stream"></param>
|
||||
/// <param name="value"></param>
|
||||
public static void WriteByteArray(this System.IO.MemoryStream stream, byte[] value)
|
||||
public static void Write(this MemoryStream stream, byte[] value)
|
||||
{
|
||||
stream.Write(value, 0, value.Length);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Helper function to write the whole content of the given byte span to a memory stream.
|
||||
/// </summary>
|
||||
/// <param name="stream"></param>
|
||||
/// <param name="value"></param>
|
||||
public static void Write(this MemoryStream stream, ReadOnlySpan<byte> value)
|
||||
{
|
||||
byte[] buffer = ArrayPool<byte>.Shared.Rent(value.Length);
|
||||
|
||||
value.CopyTo(buffer);
|
||||
stream.Write(buffer, 0, value.Length);
|
||||
|
||||
ArrayPool<byte>.Shared.Return(buffer);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
@@ -6,7 +6,7 @@ using System.Runtime.Serialization;
|
||||
|
||||
namespace S7.Net
|
||||
{
|
||||
internal class WrongNumberOfBytesException : Exception
|
||||
public class WrongNumberOfBytesException : Exception
|
||||
{
|
||||
public WrongNumberOfBytesException() : base()
|
||||
{
|
||||
@@ -27,7 +27,7 @@ namespace S7.Net
|
||||
#endif
|
||||
}
|
||||
|
||||
internal class InvalidAddressException : Exception
|
||||
public class InvalidAddressException : Exception
|
||||
{
|
||||
public InvalidAddressException() : base ()
|
||||
{
|
||||
@@ -48,7 +48,7 @@ namespace S7.Net
|
||||
#endif
|
||||
}
|
||||
|
||||
internal class InvalidVariableTypeException : Exception
|
||||
public class InvalidVariableTypeException : Exception
|
||||
{
|
||||
public InvalidVariableTypeException() : base()
|
||||
{
|
||||
@@ -69,7 +69,7 @@ namespace S7.Net
|
||||
#endif
|
||||
}
|
||||
|
||||
internal class TPKTInvalidException : Exception
|
||||
public class TPKTInvalidException : Exception
|
||||
{
|
||||
public TPKTInvalidException() : base()
|
||||
{
|
||||
@@ -90,7 +90,7 @@ namespace S7.Net
|
||||
#endif
|
||||
}
|
||||
|
||||
internal class TPDUInvalidException : Exception
|
||||
public class TPDUInvalidException : Exception
|
||||
{
|
||||
public TPDUInvalidException() : base()
|
||||
{
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
using S7.Net.Helper;
|
||||
using S7.Net.Protocol.S7;
|
||||
using S7.Net.Types;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using DateTime = S7.Net.Types.DateTime;
|
||||
@@ -10,29 +9,104 @@ namespace S7.Net
|
||||
{
|
||||
public partial class Plc
|
||||
{
|
||||
/// <summary>
|
||||
/// Creates the header to read bytes from the PLC
|
||||
/// </summary>
|
||||
/// <param name="amount"></param>
|
||||
/// <returns></returns>
|
||||
private static void BuildHeaderPackage(System.IO.MemoryStream stream, int amount = 1)
|
||||
private static void WriteTpktHeader(System.IO.MemoryStream stream, int length)
|
||||
{
|
||||
//header size = 19 bytes
|
||||
stream.WriteByteArray(new byte[] { 0x03, 0x00 });
|
||||
//complete package size
|
||||
stream.WriteByteArray(Types.Int.ToByteArray((short)(19 + (12 * amount))));
|
||||
stream.WriteByteArray(new byte[] { 0x02, 0xf0, 0x80, 0x32, 0x01, 0x00, 0x00, 0x00, 0x00 });
|
||||
//data part size
|
||||
stream.WriteByteArray(Types.Word.ToByteArray((ushort)(2 + (amount * 12))));
|
||||
stream.WriteByteArray(new byte[] { 0x00, 0x00, 0x04 });
|
||||
stream.Write(new byte[] { 0x03, 0x00 });
|
||||
stream.Write(Word.ToByteArray((ushort) length));
|
||||
}
|
||||
|
||||
private static void WriteDataHeader(System.IO.MemoryStream stream)
|
||||
{
|
||||
stream.Write(new byte[] { 0x02, 0xf0, 0x80 });
|
||||
}
|
||||
|
||||
private static void WriteS7Header(System.IO.MemoryStream stream, byte messageType, int parameterLength, int dataLength)
|
||||
{
|
||||
stream.WriteByte(0x32); // S7 protocol ID
|
||||
stream.WriteByte(messageType); // Message type
|
||||
stream.Write(new byte[] { 0x00, 0x00 }); // Reserved
|
||||
stream.Write(new byte[] { 0x00, 0x00 }); // PDU ref
|
||||
stream.Write(Word.ToByteArray((ushort) parameterLength));
|
||||
stream.Write(Word.ToByteArray((ushort) dataLength));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Creates the header to read bytes from the PLC.
|
||||
/// </summary>
|
||||
/// <param name="stream">The stream to write to.</param>
|
||||
/// <param name="amount">The number of items to read.</param>
|
||||
private static void WriteReadHeader(System.IO.MemoryStream stream, int amount = 1)
|
||||
{
|
||||
// Header size 19, 12 bytes per item
|
||||
WriteTpktHeader(stream, 19 + 12 * amount);
|
||||
WriteDataHeader(stream);
|
||||
WriteS7Header(stream, 0x01, 2 + 12 * amount, 0);
|
||||
// Function code: read request
|
||||
stream.WriteByte(0x04);
|
||||
//amount of requests
|
||||
stream.WriteByte((byte)amount);
|
||||
}
|
||||
|
||||
private static void WriteUserDataHeader(System.IO.MemoryStream stream, int parameterLength, int dataLength)
|
||||
{
|
||||
const byte s7MessageTypeUserData = 0x07;
|
||||
|
||||
WriteTpktHeader(stream, 17 + parameterLength + dataLength);
|
||||
WriteDataHeader(stream);
|
||||
WriteS7Header(stream, s7MessageTypeUserData, parameterLength, dataLength);
|
||||
}
|
||||
|
||||
private static void WriteUserDataRequest(System.IO.MemoryStream stream, byte functionGroup, byte subFunction, int dataLength)
|
||||
{
|
||||
WriteUserDataHeader(stream, 8, dataLength);
|
||||
|
||||
// Parameter
|
||||
const byte userDataMethodRequest = 0x11;
|
||||
const byte userDataTypeRequest = 0x4;
|
||||
|
||||
// Parameter head
|
||||
stream.Write(new byte[] { 0x00, 0x01, 0x12 });
|
||||
// Parameter length
|
||||
stream.WriteByte(0x04);
|
||||
// Method
|
||||
stream.WriteByte(userDataMethodRequest);
|
||||
// Type / function group
|
||||
stream.WriteByte((byte)(userDataTypeRequest << 4 | (functionGroup & 0x0f)));
|
||||
// Subfunction
|
||||
stream.WriteByte(subFunction);
|
||||
// Sequence number
|
||||
stream.WriteByte(0);
|
||||
}
|
||||
|
||||
private static void WriteSzlReadRequest(System.IO.MemoryStream stream, ushort szlId, ushort szlIndex)
|
||||
{
|
||||
// Parameter
|
||||
const byte szlFunctionGroupCpuFunctions = 0b100;
|
||||
const byte subFunctionReadSzl = 0x01;
|
||||
|
||||
WriteUserDataRequest(stream, szlFunctionGroupCpuFunctions, subFunctionReadSzl, 8);
|
||||
|
||||
// Data
|
||||
const byte success = 0xff;
|
||||
const byte transportSizeOctetString = 0x09;
|
||||
|
||||
// Return code
|
||||
stream.WriteByte(success);
|
||||
// Transport size
|
||||
stream.WriteByte(transportSizeOctetString);
|
||||
// Length
|
||||
stream.Write(Word.ToByteArray(4));
|
||||
// SZL-ID
|
||||
stream.Write(Word.ToByteArray(szlId));
|
||||
// SZL-Index
|
||||
stream.Write(Word.ToByteArray(szlIndex));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Create the bytes-package to request data from the PLC. You have to specify the memory type (dataType),
|
||||
/// the address of the memory, the address of the byte and the bytes count.
|
||||
/// </summary>
|
||||
/// <param name="stream">The stream to write the read data request to.</param>
|
||||
/// <param name="dataType">MemoryType (DB, Timer, Counter, etc.)</param>
|
||||
/// <param name="db">Address of the memory to be read</param>
|
||||
/// <param name="startByteAdr">Start address of the byte</param>
|
||||
@@ -41,7 +115,7 @@ namespace S7.Net
|
||||
private static void BuildReadDataRequestPackage(System.IO.MemoryStream stream, DataType dataType, int db, int startByteAdr, int count = 1)
|
||||
{
|
||||
//single data req = 12
|
||||
stream.WriteByteArray(new byte[] { 0x12, 0x0a, 0x10 });
|
||||
stream.Write(new byte[] { 0x12, 0x0a, 0x10 });
|
||||
switch (dataType)
|
||||
{
|
||||
case DataType.Timer:
|
||||
@@ -53,8 +127,8 @@ namespace S7.Net
|
||||
break;
|
||||
}
|
||||
|
||||
stream.WriteByteArray(Word.ToByteArray((ushort)(count)));
|
||||
stream.WriteByteArray(Word.ToByteArray((ushort)(db)));
|
||||
stream.Write(Word.ToByteArray((ushort)(count)));
|
||||
stream.Write(Word.ToByteArray((ushort)(db)));
|
||||
stream.WriteByte((byte)dataType);
|
||||
var overflow = (int)(startByteAdr * 8 / 0xffffU); // handles words with address bigger than 8191
|
||||
stream.WriteByte((byte)overflow);
|
||||
@@ -62,10 +136,10 @@ namespace S7.Net
|
||||
{
|
||||
case DataType.Timer:
|
||||
case DataType.Counter:
|
||||
stream.WriteByteArray(Types.Word.ToByteArray((ushort)(startByteAdr)));
|
||||
stream.Write(Word.ToByteArray((ushort)(startByteAdr)));
|
||||
break;
|
||||
default:
|
||||
stream.WriteByteArray(Types.Word.ToByteArray((ushort)((startByteAdr) * 8)));
|
||||
stream.Write(Word.ToByteArray((ushort)((startByteAdr) * 8)));
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -168,6 +242,24 @@ namespace S7.Net
|
||||
{
|
||||
return DateTimeLong.ToArray(bytes);
|
||||
}
|
||||
case VarType.Time:
|
||||
if (varCount == 1)
|
||||
{
|
||||
return TimeSpan.FromByteArray(bytes);
|
||||
}
|
||||
else
|
||||
{
|
||||
return TimeSpan.ToArray(bytes);
|
||||
}
|
||||
case VarType.Date:
|
||||
if (varCount == 1)
|
||||
{
|
||||
return Date.FromByteArray(bytes);
|
||||
}
|
||||
else
|
||||
{
|
||||
return Date.ToArray(bytes);
|
||||
}
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
@@ -197,10 +289,12 @@ namespace S7.Net
|
||||
case VarType.Timer:
|
||||
case VarType.Int:
|
||||
case VarType.Counter:
|
||||
case VarType.Date:
|
||||
return varCount * 2;
|
||||
case VarType.DWord:
|
||||
case VarType.DInt:
|
||||
case VarType.Real:
|
||||
case VarType.Time:
|
||||
return varCount * 4;
|
||||
case VarType.LReal:
|
||||
case VarType.DateTime:
|
||||
@@ -253,7 +347,7 @@ namespace S7.Net
|
||||
int packageSize = 19 + (dataItems.Count * 12);
|
||||
var package = new System.IO.MemoryStream(packageSize);
|
||||
|
||||
BuildHeaderPackage(package, dataItems.Count);
|
||||
WriteReadHeader(package, dataItems.Count);
|
||||
|
||||
foreach (var dataItem in dataItems)
|
||||
{
|
||||
@@ -262,5 +356,15 @@ namespace S7.Net
|
||||
|
||||
return package.ToArray();
|
||||
}
|
||||
|
||||
private static byte[] BuildSzlReadRequestPackage(ushort szlId, ushort szlIndex)
|
||||
{
|
||||
var stream = new System.IO.MemoryStream();
|
||||
|
||||
WriteSzlReadRequest(stream, szlId, szlIndex);
|
||||
stream.SetLength(stream.Position);
|
||||
|
||||
return stream.ToArray();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
92
S7.Net/Plc.Clock.cs
Normal file
92
S7.Net/Plc.Clock.cs
Normal file
@@ -0,0 +1,92 @@
|
||||
using System;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using S7.Net.Helper;
|
||||
using S7.Net.Types;
|
||||
using DateTime = System.DateTime;
|
||||
|
||||
namespace S7.Net;
|
||||
|
||||
partial class Plc
|
||||
{
|
||||
private const byte SzlFunctionGroupTimers = 0x07;
|
||||
private const byte SzlSubFunctionReadClock = 0x01;
|
||||
private const byte SzlSubFunctionWriteClock = 0x02;
|
||||
private const byte TransportSizeOctetString = 0x09;
|
||||
private const int PduErrOffset = 20;
|
||||
private const int UserDataResultOffset = PduErrOffset + 2;
|
||||
|
||||
/// <summary>
|
||||
/// The length in bytes of DateTime stored in the PLC.
|
||||
/// </summary>
|
||||
private const int DateTimeLength = 10;
|
||||
|
||||
private static byte[] BuildClockReadRequest()
|
||||
{
|
||||
var stream = new MemoryStream();
|
||||
|
||||
WriteUserDataRequest(stream, SzlFunctionGroupTimers, SzlSubFunctionReadClock, 4);
|
||||
stream.Write(new byte[] { 0x0a, 0x00, 0x00, 0x00 });
|
||||
|
||||
stream.SetLength(stream.Position);
|
||||
return stream.ToArray();
|
||||
}
|
||||
|
||||
private static DateTime ParseClockReadResponse(byte[] message)
|
||||
{
|
||||
const int udLenOffset = UserDataResultOffset + 2;
|
||||
const int udValueOffset = udLenOffset + 2;
|
||||
const int dateTimeSkip = 2;
|
||||
|
||||
AssertPduResult(message);
|
||||
AssertUserDataResult(message, 0xff);
|
||||
|
||||
var len = Word.FromByteArray(message.Skip(udLenOffset).Take(2).ToArray());
|
||||
if (len != DateTimeLength)
|
||||
{
|
||||
throw new Exception($"Unexpected response length {len}, expected {DateTimeLength}.");
|
||||
}
|
||||
|
||||
// Skip first 2 bytes from date time value because DateTime.FromByteArray doesn't parse them.
|
||||
return Types.DateTime.FromByteArray(message.Skip(udValueOffset + dateTimeSkip)
|
||||
.Take(DateTimeLength - dateTimeSkip).ToArray());
|
||||
}
|
||||
|
||||
private static byte[] BuildClockWriteRequest(DateTime value)
|
||||
{
|
||||
var stream = new MemoryStream();
|
||||
|
||||
WriteUserDataRequest(stream, SzlFunctionGroupTimers, SzlSubFunctionWriteClock, 14);
|
||||
stream.Write(new byte[] { 0xff, TransportSizeOctetString, 0x00, DateTimeLength });
|
||||
// Start of DateTime value, DateTime.ToByteArray only serializes the final 8 bytes
|
||||
stream.Write(new byte[] { 0x00, 0x19 });
|
||||
stream.Write(Types.DateTime.ToByteArray(value));
|
||||
|
||||
stream.SetLength(stream.Position);
|
||||
return stream.ToArray();
|
||||
}
|
||||
|
||||
private static void ParseClockWriteResponse(byte[] message)
|
||||
{
|
||||
AssertPduResult(message);
|
||||
AssertUserDataResult(message, 0x0a);
|
||||
}
|
||||
|
||||
private static void AssertPduResult(byte[] message)
|
||||
{
|
||||
var pduErr = Word.FromByteArray(message.Skip(PduErrOffset).Take(2).ToArray());
|
||||
if (pduErr != 0)
|
||||
{
|
||||
throw new Exception($"Response from PLC indicates error 0x{pduErr:X4}.");
|
||||
}
|
||||
}
|
||||
|
||||
private static void AssertUserDataResult(byte[] message, byte expected)
|
||||
{
|
||||
var dtResult = message[UserDataResultOffset];
|
||||
if (dtResult != expected)
|
||||
{
|
||||
throw new Exception($"Response from PLC was 0x{dtResult:X2}, expected 0x{expected:X2}.");
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -25,7 +25,7 @@ namespace S7.Net
|
||||
/// <returns>A task that represents the asynchronous open operation.</returns>
|
||||
public async Task OpenAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
var stream = await ConnectAsync().ConfigureAwait(false);
|
||||
var stream = await ConnectAsync(cancellationToken).ConfigureAwait(false);
|
||||
try
|
||||
{
|
||||
await queue.Enqueue(async () =>
|
||||
@@ -44,11 +44,16 @@ namespace S7.Net
|
||||
}
|
||||
}
|
||||
|
||||
private async Task<NetworkStream> ConnectAsync()
|
||||
private async Task<NetworkStream> ConnectAsync(CancellationToken cancellationToken)
|
||||
{
|
||||
tcpClient = new TcpClient();
|
||||
ConfigureConnection();
|
||||
|
||||
#if NET5_0_OR_GREATER
|
||||
await tcpClient.ConnectAsync(IP, Port, cancellationToken).ConfigureAwait(false);
|
||||
#else
|
||||
await tcpClient.ConnectAsync(IP, Port).ConfigureAwait(false);
|
||||
#endif
|
||||
return tcpClient.GetStream();
|
||||
}
|
||||
|
||||
@@ -90,7 +95,6 @@ namespace S7.Net
|
||||
MaxPDUSize = s7data[18] * 256 + s7data[19];
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Reads a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
/// If the read was not successful, check LastErrorCode or LastErrorString.
|
||||
@@ -105,16 +109,34 @@ namespace S7.Net
|
||||
public async Task<byte[]> ReadBytesAsync(DataType dataType, int db, int startByteAdr, int count, CancellationToken cancellationToken = default)
|
||||
{
|
||||
var resultBytes = new byte[count];
|
||||
|
||||
await ReadBytesAsync(resultBytes, dataType, db, startByteAdr, cancellationToken).ConfigureAwait(false);
|
||||
|
||||
return resultBytes;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Reads a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
/// If the read was not successful, check LastErrorCode or LastErrorString.
|
||||
/// </summary>
|
||||
/// <param name="buffer">Buffer to receive the read bytes. The <see cref="Memory{T}.Length"/> determines the number of bytes to read.</param>
|
||||
/// <param name="dataType">Data type of the memory area, can be DB, Timer, Counter, Merker(Memory), Input, Output.</param>
|
||||
/// <param name="db">Address of the memory area (if you want to read DB1, this is set to 1). This must be set also for other memory area types: counters, timers,etc.</param>
|
||||
/// <param name="startByteAdr">Start byte address. If you want to read DB1.DBW200, this is 200.</param>
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>Returns the bytes in an array</returns>
|
||||
public async Task ReadBytesAsync(Memory<byte> buffer, DataType dataType, int db, int startByteAdr, CancellationToken cancellationToken = default)
|
||||
{
|
||||
int index = 0;
|
||||
while (count > 0)
|
||||
while (buffer.Length > 0)
|
||||
{
|
||||
//This works up to MaxPDUSize-1 on SNAP7. But not MaxPDUSize-0.
|
||||
var maxToRead = Math.Min(count, MaxPDUSize - 18);
|
||||
await ReadBytesWithSingleRequestAsync(dataType, db, startByteAdr + index, resultBytes, index, maxToRead, cancellationToken).ConfigureAwait(false);
|
||||
count -= maxToRead;
|
||||
var maxToRead = Math.Min(buffer.Length, MaxPDUSize - 18);
|
||||
await ReadBytesWithSingleRequestAsync(dataType, db, startByteAdr + index, buffer.Slice(0, maxToRead), cancellationToken).ConfigureAwait(false);
|
||||
buffer = buffer.Slice(maxToRead);
|
||||
index += maxToRead;
|
||||
}
|
||||
return resultBytes;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -290,6 +312,48 @@ namespace S7.Net
|
||||
return dataItems;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Read the PLC clock value.
|
||||
/// </summary>
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>A task that represents the asynchronous operation, with it's result set to the current PLC time on completion.</returns>
|
||||
public async Task<System.DateTime> ReadClockAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
var request = BuildClockReadRequest();
|
||||
var response = await RequestTsduAsync(request, cancellationToken);
|
||||
|
||||
return ParseClockReadResponse(response);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Write the PLC clock value.
|
||||
/// </summary>
|
||||
/// <param name="value">The date and time to set the PLC clock to</param>
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>A task that represents the asynchronous operation.</returns>
|
||||
public async Task WriteClockAsync(System.DateTime value, CancellationToken cancellationToken = default)
|
||||
{
|
||||
var request = BuildClockWriteRequest(value);
|
||||
var response = await RequestTsduAsync(request, cancellationToken);
|
||||
|
||||
ParseClockWriteResponse(response);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Read the current status from the PLC. A value of 0x08 indicates the PLC is in run status, regardless of the PLC type.
|
||||
/// </summary>
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>A task that represents the asynchronous operation, with it's result set to the current PLC status on completion.</returns>
|
||||
public async Task<byte> ReadStatusAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
var dataToSend = BuildSzlReadRequestPackage(0x0424, 0);
|
||||
var s7data = await RequestTsduAsync(dataToSend, cancellationToken);
|
||||
|
||||
return (byte) (s7data[37] & 0x0f);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Write a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
@@ -302,15 +366,30 @@ namespace S7.Net
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>A task that represents the asynchronous write operation.</returns>
|
||||
public async Task WriteBytesAsync(DataType dataType, int db, int startByteAdr, byte[] value, CancellationToken cancellationToken = default)
|
||||
public Task WriteBytesAsync(DataType dataType, int db, int startByteAdr, byte[] value, CancellationToken cancellationToken = default)
|
||||
{
|
||||
return WriteBytesAsync(dataType, db, startByteAdr, value.AsMemory(), cancellationToken);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Write a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
/// If the write was not successful, check LastErrorCode or LastErrorString.
|
||||
/// </summary>
|
||||
/// <param name="dataType">Data type of the memory area, can be DB, Timer, Counter, Merker(Memory), Input, Output.</param>
|
||||
/// <param name="db">Address of the memory area (if you want to read DB1, this is set to 1). This must be set also for other memory area types: counters, timers,etc.</param>
|
||||
/// <param name="startByteAdr">Start byte address. If you want to write DB1.DBW200, this is 200.</param>
|
||||
/// <param name="value">Bytes to write. If more than 200, multiple requests will be made.</param>
|
||||
/// <param name="cancellationToken">The token to monitor for cancellation requests. The default value is None.
|
||||
/// Please note that cancellation is advisory/cooperative and will not lead to immediate cancellation in all cases.</param>
|
||||
/// <returns>A task that represents the asynchronous write operation.</returns>
|
||||
public async Task WriteBytesAsync(DataType dataType, int db, int startByteAdr, ReadOnlyMemory<byte> value, CancellationToken cancellationToken = default)
|
||||
{
|
||||
int localIndex = 0;
|
||||
int count = value.Length;
|
||||
while (count > 0)
|
||||
while (value.Length > 0)
|
||||
{
|
||||
var maxToWrite = (int)Math.Min(count, MaxPDUSize - 35);
|
||||
await WriteBytesWithASingleRequestAsync(dataType, db, startByteAdr + localIndex, value, localIndex, maxToWrite, cancellationToken).ConfigureAwait(false);
|
||||
count -= maxToWrite;
|
||||
var maxToWrite = (int)Math.Min(value.Length, MaxPDUSize - 35);
|
||||
await WriteBytesWithASingleRequestAsync(dataType, db, startByteAdr + localIndex, value.Slice(0, maxToWrite), cancellationToken).ConfigureAwait(false);
|
||||
value = value.Slice(maxToWrite);
|
||||
localIndex += maxToWrite;
|
||||
}
|
||||
}
|
||||
@@ -392,7 +471,6 @@ namespace S7.Net
|
||||
|
||||
/// <summary>
|
||||
/// Writes a single variable from the PLC, takes in input strings like "DB1.DBX0.0", "DB20.DBD200", "MB20", "T45", etc.
|
||||
/// If the write was not successful, check <see cref="LastErrorCode"/> or <see cref="LastErrorString"/>.
|
||||
/// </summary>
|
||||
/// <param name="variable">Input strings like "DB1.DBX0.0", "DB20.DBD200", "MB20", "T45", etc.</param>
|
||||
/// <param name="value">Value to be written to the PLC</param>
|
||||
@@ -436,14 +514,14 @@ namespace S7.Net
|
||||
await WriteBytesAsync(DataType.DataBlock, db, startByteAdr, bytes, cancellationToken).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
private async Task ReadBytesWithSingleRequestAsync(DataType dataType, int db, int startByteAdr, byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
||||
private async Task ReadBytesWithSingleRequestAsync(DataType dataType, int db, int startByteAdr, Memory<byte> buffer, CancellationToken cancellationToken)
|
||||
{
|
||||
var dataToSend = BuildReadRequestPackage(new[] { new DataItemAddress(dataType, db, startByteAdr, count) });
|
||||
var dataToSend = BuildReadRequestPackage(new[] { new DataItemAddress(dataType, db, startByteAdr, buffer.Length) });
|
||||
|
||||
var s7data = await RequestTsduAsync(dataToSend, cancellationToken);
|
||||
AssertReadResponse(s7data, count);
|
||||
AssertReadResponse(s7data, buffer.Length);
|
||||
|
||||
Array.Copy(s7data, 18, buffer, offset, count);
|
||||
s7data.AsSpan(18, buffer.Length).CopyTo(buffer.Span);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -471,12 +549,13 @@ namespace S7.Net
|
||||
/// <param name="db">Address of the memory area (if you want to read DB1, this is set to 1). This must be set also for other memory area types: counters, timers,etc.</param>
|
||||
/// <param name="startByteAdr">Start byte address. If you want to read DB1.DBW200, this is 200.</param>
|
||||
/// <param name="value">Bytes to write. The lenght of this parameter can't be higher than 200. If you need more, use recursion.</param>
|
||||
/// <param name="cancellationToken">A cancellation token that can be used to cancel the asynchronous operation.</param>
|
||||
/// <returns>A task that represents the asynchronous write operation.</returns>
|
||||
private async Task WriteBytesWithASingleRequestAsync(DataType dataType, int db, int startByteAdr, byte[] value, int dataOffset, int count, CancellationToken cancellationToken)
|
||||
private async Task WriteBytesWithASingleRequestAsync(DataType dataType, int db, int startByteAdr, ReadOnlyMemory<byte> value, CancellationToken cancellationToken)
|
||||
{
|
||||
try
|
||||
{
|
||||
var dataToSend = BuildWriteBytesPackage(dataType, db, startByteAdr, value, dataOffset, count);
|
||||
var dataToSend = BuildWriteBytesPackage(dataType, db, startByteAdr, value.Span);
|
||||
var s7data = await RequestTsduAsync(dataToSend, cancellationToken).ConfigureAwait(false);
|
||||
|
||||
ValidateResponseCode((ReadWriteErrorCode)s7data[14]);
|
||||
|
||||
@@ -39,16 +39,32 @@ namespace S7.Net
|
||||
public byte[] ReadBytes(DataType dataType, int db, int startByteAdr, int count)
|
||||
{
|
||||
var result = new byte[count];
|
||||
|
||||
ReadBytes(result, dataType, db, startByteAdr);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Reads a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
/// If the read was not successful, check LastErrorCode or LastErrorString.
|
||||
/// </summary>
|
||||
/// <param name="buffer">Buffer to receive the read bytes. The <see cref="Span{T}.Length"/> determines the number of bytes to read.</param>
|
||||
/// <param name="dataType">Data type of the memory area, can be DB, Timer, Counter, Merker(Memory), Input, Output.</param>
|
||||
/// <param name="db">Address of the memory area (if you want to read DB1, this is set to 1). This must be set also for other memory area types: counters, timers,etc.</param>
|
||||
/// <param name="startByteAdr">Start byte address. If you want to read DB1.DBW200, this is 200.</param>
|
||||
/// <returns>Returns the bytes in an array</returns>
|
||||
public void ReadBytes(Span<byte> buffer, DataType dataType, int db, int startByteAdr)
|
||||
{
|
||||
int index = 0;
|
||||
while (count > 0)
|
||||
while (buffer.Length > 0)
|
||||
{
|
||||
//This works up to MaxPDUSize-1 on SNAP7. But not MaxPDUSize-0.
|
||||
var maxToRead = Math.Min(count, MaxPDUSize - 18);
|
||||
ReadBytesWithSingleRequest(dataType, db, startByteAdr + index, result, index, maxToRead);
|
||||
count -= maxToRead;
|
||||
var maxToRead = Math.Min(buffer.Length, MaxPDUSize - 18);
|
||||
ReadBytesWithSingleRequest(dataType, db, startByteAdr + index, buffer.Slice(0, maxToRead));
|
||||
buffer = buffer.Slice(maxToRead);
|
||||
index += maxToRead;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -111,7 +127,6 @@ namespace S7.Net
|
||||
return ReadStruct(typeof(T), db, startByteAdr) as T?;
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Reads all the bytes needed to fill a class in C#, starting from a certain address, and set all the properties values to the value that are read from the PLC.
|
||||
/// This reads only properties, it doesn't read private variable or public variable without {get;set;} specified.
|
||||
@@ -178,17 +193,29 @@ namespace S7.Net
|
||||
/// <param name="startByteAdr">Start byte address. If you want to write DB1.DBW200, this is 200.</param>
|
||||
/// <param name="value">Bytes to write. If more than 200, multiple requests will be made.</param>
|
||||
public void WriteBytes(DataType dataType, int db, int startByteAdr, byte[] value)
|
||||
{
|
||||
WriteBytes(dataType, db, startByteAdr, value.AsSpan());
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Write a number of bytes from a DB starting from a specified index. This handles more than 200 bytes with multiple requests.
|
||||
/// If the write was not successful, check LastErrorCode or LastErrorString.
|
||||
/// </summary>
|
||||
/// <param name="dataType">Data type of the memory area, can be DB, Timer, Counter, Merker(Memory), Input, Output.</param>
|
||||
/// <param name="db">Address of the memory area (if you want to read DB1, this is set to 1). This must be set also for other memory area types: counters, timers,etc.</param>
|
||||
/// <param name="startByteAdr">Start byte address. If you want to write DB1.DBW200, this is 200.</param>
|
||||
/// <param name="value">Bytes to write. If more than 200, multiple requests will be made.</param>
|
||||
public void WriteBytes(DataType dataType, int db, int startByteAdr, ReadOnlySpan<byte> value)
|
||||
{
|
||||
int localIndex = 0;
|
||||
int count = value.Length;
|
||||
while (count > 0)
|
||||
while (value.Length > 0)
|
||||
{
|
||||
//TODO: Figure out how to use MaxPDUSize here
|
||||
//Snap7 seems to choke on PDU sizes above 256 even if snap7
|
||||
//replies with bigger PDU size in connection setup.
|
||||
var maxToWrite = Math.Min(count, MaxPDUSize - 28);//TODO tested only when the MaxPDUSize is 480
|
||||
WriteBytesWithASingleRequest(dataType, db, startByteAdr + localIndex, value, localIndex, maxToWrite);
|
||||
count -= maxToWrite;
|
||||
var maxToWrite = Math.Min(value.Length, MaxPDUSize - 28);//TODO tested only when the MaxPDUSize is 480
|
||||
WriteBytesWithASingleRequest(dataType, db, startByteAdr + localIndex, value.Slice(0, maxToWrite));
|
||||
value = value.Slice(maxToWrite);
|
||||
localIndex += maxToWrite;
|
||||
}
|
||||
}
|
||||
@@ -262,7 +289,6 @@ namespace S7.Net
|
||||
|
||||
/// <summary>
|
||||
/// Writes a single variable from the PLC, takes in input strings like "DB1.DBX0.0", "DB20.DBD200", "MB20", "T45", etc.
|
||||
/// If the write was not successful, check <see cref="LastErrorCode"/> or <see cref="LastErrorString"/>.
|
||||
/// </summary>
|
||||
/// <param name="variable">Input strings like "DB1.DBX0.0", "DB20.DBD200", "MB20", "T45", etc.</param>
|
||||
/// <param name="value">Value to be written to the PLC</param>
|
||||
@@ -294,22 +320,22 @@ namespace S7.Net
|
||||
WriteClassAsync(classValue, db, startByteAdr).GetAwaiter().GetResult();
|
||||
}
|
||||
|
||||
private void ReadBytesWithSingleRequest(DataType dataType, int db, int startByteAdr, byte[] buffer, int offset, int count)
|
||||
private void ReadBytesWithSingleRequest(DataType dataType, int db, int startByteAdr, Span<byte> buffer)
|
||||
{
|
||||
try
|
||||
{
|
||||
// first create the header
|
||||
int packageSize = 19 + 12; // 19 header + 12 for 1 request
|
||||
var package = new System.IO.MemoryStream(packageSize);
|
||||
BuildHeaderPackage(package);
|
||||
const int packageSize = 19 + 12; // 19 header + 12 for 1 request
|
||||
var dataToSend = new byte[packageSize];
|
||||
var package = new MemoryStream(dataToSend);
|
||||
WriteReadHeader(package);
|
||||
// package.Add(0x02); // datenart
|
||||
BuildReadDataRequestPackage(package, dataType, db, startByteAdr, count);
|
||||
BuildReadDataRequestPackage(package, dataType, db, startByteAdr, buffer.Length);
|
||||
|
||||
var dataToSend = package.ToArray();
|
||||
var s7data = RequestTsdu(dataToSend);
|
||||
AssertReadResponse(s7data, count);
|
||||
AssertReadResponse(s7data, buffer.Length);
|
||||
|
||||
Array.Copy(s7data, 18, buffer, offset, count);
|
||||
s7data.AsSpan(18, buffer.Length).CopyTo(buffer);
|
||||
}
|
||||
catch (Exception exc)
|
||||
{
|
||||
@@ -326,7 +352,6 @@ namespace S7.Net
|
||||
{
|
||||
AssertPduSizeForWrite(dataItems);
|
||||
|
||||
|
||||
var message = new ByteArray();
|
||||
var length = S7WriteMultiple.CreateRequest(message, dataItems);
|
||||
var response = RequestTsdu(message.Array, 0, length);
|
||||
@@ -334,11 +359,11 @@ namespace S7.Net
|
||||
S7WriteMultiple.ParseResponse(response, response.Length, dataItems);
|
||||
}
|
||||
|
||||
private void WriteBytesWithASingleRequest(DataType dataType, int db, int startByteAdr, byte[] value, int dataOffset, int count)
|
||||
private void WriteBytesWithASingleRequest(DataType dataType, int db, int startByteAdr, ReadOnlySpan<byte> value)
|
||||
{
|
||||
try
|
||||
{
|
||||
var dataToSend = BuildWriteBytesPackage(dataType, db, startByteAdr, value, dataOffset, count);
|
||||
var dataToSend = BuildWriteBytesPackage(dataType, db, startByteAdr, value);
|
||||
var s7data = RequestTsdu(dataToSend);
|
||||
|
||||
ValidateResponseCode((ReadWriteErrorCode)s7data[14]);
|
||||
@@ -349,35 +374,37 @@ namespace S7.Net
|
||||
}
|
||||
}
|
||||
|
||||
private byte[] BuildWriteBytesPackage(DataType dataType, int db, int startByteAdr, byte[] value, int dataOffset, int count)
|
||||
private byte[] BuildWriteBytesPackage(DataType dataType, int db, int startByteAdr, ReadOnlySpan<byte> value)
|
||||
{
|
||||
int varCount = count;
|
||||
int varCount = value.Length;
|
||||
// first create the header
|
||||
int packageSize = 35 + varCount;
|
||||
var package = new MemoryStream(new byte[packageSize]);
|
||||
var packageData = new byte[packageSize];
|
||||
var package = new MemoryStream(packageData);
|
||||
|
||||
package.WriteByte(3);
|
||||
package.WriteByte(0);
|
||||
//complete package size
|
||||
package.WriteByteArray(Int.ToByteArray((short)packageSize));
|
||||
package.WriteByteArray(new byte[] { 2, 0xf0, 0x80, 0x32, 1, 0, 0 });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount - 1)));
|
||||
package.WriteByteArray(new byte[] { 0, 0x0e });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount + 4)));
|
||||
package.WriteByteArray(new byte[] { 0x05, 0x01, 0x12, 0x0a, 0x10, 0x02 });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)varCount));
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(db)));
|
||||
package.Write(Int.ToByteArray((short)packageSize));
|
||||
// This overload doesn't allocate the byte array, it refers to assembly's static data segment
|
||||
package.Write(new byte[] { 2, 0xf0, 0x80, 0x32, 1, 0, 0 });
|
||||
package.Write(Word.ToByteArray((ushort)(varCount - 1)));
|
||||
package.Write(new byte[] { 0, 0x0e });
|
||||
package.Write(Word.ToByteArray((ushort)(varCount + 4)));
|
||||
package.Write(new byte[] { 0x05, 0x01, 0x12, 0x0a, 0x10, 0x02 });
|
||||
package.Write(Word.ToByteArray((ushort)varCount));
|
||||
package.Write(Word.ToByteArray((ushort)(db)));
|
||||
package.WriteByte((byte)dataType);
|
||||
var overflow = (int)(startByteAdr * 8 / 0xffffU); // handles words with address bigger than 8191
|
||||
package.WriteByte((byte)overflow);
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(startByteAdr * 8)));
|
||||
package.WriteByteArray(new byte[] { 0, 4 });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount * 8)));
|
||||
package.Write(Word.ToByteArray((ushort)(startByteAdr * 8)));
|
||||
package.Write(new byte[] { 0, 4 });
|
||||
package.Write(Word.ToByteArray((ushort)(varCount * 8)));
|
||||
|
||||
// now join the header and the data
|
||||
package.Write(value, dataOffset, count);
|
||||
package.Write(value);
|
||||
|
||||
return package.ToArray();
|
||||
return packageData;
|
||||
}
|
||||
|
||||
private byte[] BuildWriteBitPackage(DataType dataType, int db, int startByteAdr, bool bitValue, int bitAdr)
|
||||
@@ -386,33 +413,33 @@ namespace S7.Net
|
||||
int varCount = 1;
|
||||
// first create the header
|
||||
int packageSize = 35 + varCount;
|
||||
var package = new MemoryStream(new byte[packageSize]);
|
||||
var packageData = new byte[packageSize];
|
||||
var package = new MemoryStream(packageData);
|
||||
|
||||
package.WriteByte(3);
|
||||
package.WriteByte(0);
|
||||
//complete package size
|
||||
package.WriteByteArray(Int.ToByteArray((short)packageSize));
|
||||
package.WriteByteArray(new byte[] { 2, 0xf0, 0x80, 0x32, 1, 0, 0 });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount - 1)));
|
||||
package.WriteByteArray(new byte[] { 0, 0x0e });
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount + 4)));
|
||||
package.WriteByteArray(new byte[] { 0x05, 0x01, 0x12, 0x0a, 0x10, 0x01 }); //ending 0x01 is used for writing a sinlge bit
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)varCount));
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(db)));
|
||||
package.Write(Int.ToByteArray((short)packageSize));
|
||||
package.Write(new byte[] { 2, 0xf0, 0x80, 0x32, 1, 0, 0 });
|
||||
package.Write(Word.ToByteArray((ushort)(varCount - 1)));
|
||||
package.Write(new byte[] { 0, 0x0e });
|
||||
package.Write(Word.ToByteArray((ushort)(varCount + 4)));
|
||||
package.Write(new byte[] { 0x05, 0x01, 0x12, 0x0a, 0x10, 0x01 }); //ending 0x01 is used for writing a sinlge bit
|
||||
package.Write(Word.ToByteArray((ushort)varCount));
|
||||
package.Write(Word.ToByteArray((ushort)(db)));
|
||||
package.WriteByte((byte)dataType);
|
||||
var overflow = (int)(startByteAdr * 8 / 0xffffU); // handles words with address bigger than 8191
|
||||
package.WriteByte((byte)overflow);
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(startByteAdr * 8 + bitAdr)));
|
||||
package.WriteByteArray(new byte[] { 0, 0x03 }); //ending 0x03 is used for writing a sinlge bit
|
||||
package.WriteByteArray(Word.ToByteArray((ushort)(varCount)));
|
||||
package.Write(Word.ToByteArray((ushort)(startByteAdr * 8 + bitAdr)));
|
||||
package.Write(new byte[] { 0, 0x03 }); //ending 0x03 is used for writing a sinlge bit
|
||||
package.Write(Word.ToByteArray((ushort)(varCount)));
|
||||
|
||||
// now join the header and the data
|
||||
package.WriteByteArray(value);
|
||||
package.Write(value);
|
||||
|
||||
return package.ToArray();
|
||||
return packageData;
|
||||
}
|
||||
|
||||
|
||||
private void WriteBitWithASingleRequest(DataType dataType, int db, int startByteAdr, int bitAdr, bool bitValue)
|
||||
{
|
||||
try
|
||||
@@ -444,16 +471,16 @@ namespace S7.Net
|
||||
{
|
||||
// first create the header
|
||||
int packageSize = 19 + (dataItems.Count * 12);
|
||||
var package = new System.IO.MemoryStream(packageSize);
|
||||
BuildHeaderPackage(package, dataItems.Count);
|
||||
var dataToSend = new byte[packageSize];
|
||||
var package = new MemoryStream(dataToSend);
|
||||
WriteReadHeader(package, dataItems.Count);
|
||||
// package.Add(0x02); // datenart
|
||||
foreach (var dataItem in dataItems)
|
||||
{
|
||||
BuildReadDataRequestPackage(package, dataItem.DataType, dataItem.DB, dataItem.StartByteAdr, VarTypeToByteLength(dataItem.VarType, dataItem.Count));
|
||||
}
|
||||
|
||||
var dataToSend = package.ToArray();
|
||||
var s7data = RequestTsdu(dataToSend);
|
||||
byte[] s7data = RequestTsdu(dataToSend);
|
||||
|
||||
ValidateResponseCode((ReadWriteErrorCode)s7data[14]);
|
||||
|
||||
@@ -465,6 +492,42 @@ namespace S7.Net
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Read the PLC clock value.
|
||||
/// </summary>
|
||||
/// <returns>The current PLC time.</returns>
|
||||
public System.DateTime ReadClock()
|
||||
{
|
||||
var request = BuildClockReadRequest();
|
||||
var response = RequestTsdu(request);
|
||||
|
||||
return ParseClockReadResponse(response);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Write the PLC clock value.
|
||||
/// </summary>
|
||||
/// <param name="value">The date and time to set the PLC clock to.</param>
|
||||
public void WriteClock(System.DateTime value)
|
||||
{
|
||||
var request = BuildClockWriteRequest(value);
|
||||
var response = RequestTsdu(request);
|
||||
|
||||
ParseClockWriteResponse(response);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Read the current status from the PLC. A value of 0x08 indicates the PLC is in run status, regardless of the PLC type.
|
||||
/// </summary>
|
||||
/// <returns>The current PLC status.</returns>
|
||||
public byte ReadStatus()
|
||||
{
|
||||
var dataToSend = BuildSzlReadRequestPackage(0x0424, 0);
|
||||
var s7data = RequestTsdu(dataToSend);
|
||||
|
||||
return (byte) (s7data[37] & 0x0f);
|
||||
}
|
||||
|
||||
private byte[] RequestTsdu(byte[] requestData) => RequestTsdu(requestData, 0, requestData.Length);
|
||||
|
||||
private byte[] RequestTsdu(byte[] requestData, int offset, int length)
|
||||
|
||||
@@ -26,6 +26,11 @@ namespace S7.Net.Protocol
|
||||
_ => Types.String.ToByteArray(s, dataItem.Count)
|
||||
};
|
||||
|
||||
if (dataItem.VarType == VarType.Date)
|
||||
{
|
||||
return Date.ToByteArray((System.DateTime)dataItem.Value);
|
||||
}
|
||||
|
||||
return SerializeValue(dataItem.Value);
|
||||
}
|
||||
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFrameworks>net452;netstandard2.0;netstandard1.3</TargetFrameworks>
|
||||
<TargetFrameworks>net452;net462;netstandard2.0;netstandard1.3;net5.0;net6.0;net7.0</TargetFrameworks>
|
||||
<SignAssembly>true</SignAssembly>
|
||||
<AssemblyOriginatorKeyFile>Properties\S7.Net.snk</AssemblyOriginatorKeyFile>
|
||||
<InternalsVisibleTo>S7.Net.UnitTest</InternalsVisibleTo>
|
||||
@@ -15,17 +15,22 @@
|
||||
<RepositoryType>git</RepositoryType>
|
||||
<PackageTags>PLC Siemens Communication S7</PackageTags>
|
||||
<Copyright>Derek Heiser 2015</Copyright>
|
||||
<LangVersion>8.0</LangVersion>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<Nullable>Enable</Nullable>
|
||||
<DebugType>portable</DebugType>
|
||||
<IncludeSymbols>true</IncludeSymbols>
|
||||
<SymbolPackageFormat>snupkg</SymbolPackageFormat>
|
||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||
<NoWarn>$(NoWarn);CS1591;NETSDK1138</NoWarn>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup Condition="'$(TargetFramework)' == 'net452' Or '$(TargetFramework)' == 'netstandard2.0' ">
|
||||
<PropertyGroup Condition="'$(TargetFramework)' == 'net452' Or '$(TargetFramework)' == 'net462' Or '$(TargetFramework)' == 'netstandard2.0' ">
|
||||
<DefineConstants>NET_FULL</DefineConstants>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup Condition="'$(TargetFramework)' != 'net5.0' And '$(TargetFramework)' != 'net6.0' And '$(TargetFramework)' != 'net7.0'">
|
||||
<PackageReference Include="System.Memory" Version="4.5.5" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.SourceLink.GitHub" Version="1.0.0" PrivateAssets="All" />
|
||||
|
||||
@@ -39,6 +39,7 @@ namespace S7.Net
|
||||
/// <param name="buffer">the buffer to read into</param>
|
||||
/// <param name="offset">the offset in the buffer to read into</param>
|
||||
/// <param name="count">the amount of bytes to read into the buffer</param>
|
||||
/// <param name="cancellationToken">A cancellation token that can be used to cancel the asynchronous operation.</param>
|
||||
/// <returns>returns the amount of read bytes</returns>
|
||||
public static async Task<int> ReadExactAsync(this Stream stream, byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
||||
{
|
||||
|
||||
@@ -29,6 +29,7 @@ namespace S7.Net
|
||||
/// Reads a TPKT from the socket Async
|
||||
/// </summary>
|
||||
/// <param name="stream">The stream to read from</param>
|
||||
/// <param name="cancellationToken">A cancellation token that can be used to cancel the asynchronous operation.</param>
|
||||
/// <returns>Task TPKT Instace</returns>
|
||||
public static async Task<TPKT> ReadAsync(Stream stream, CancellationToken cancellationToken)
|
||||
{
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Reflection;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace S7.Net.Types
|
||||
{
|
||||
@@ -25,7 +26,7 @@ namespace S7.Net.Types
|
||||
|
||||
}
|
||||
|
||||
private static double GetIncreasedNumberOfBytes(double numBytes, Type type)
|
||||
private static double GetIncreasedNumberOfBytes(double numBytes, Type type, PropertyInfo? propertyInfo)
|
||||
{
|
||||
switch (type.Name)
|
||||
{
|
||||
@@ -38,32 +39,33 @@ namespace S7.Net.Types
|
||||
break;
|
||||
case "Int16":
|
||||
case "UInt16":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
numBytes += 2;
|
||||
break;
|
||||
case "Int32":
|
||||
case "UInt32":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
numBytes += 4;
|
||||
break;
|
||||
case "Single":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
numBytes += 4;
|
||||
break;
|
||||
case "Double":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
numBytes += 8;
|
||||
break;
|
||||
case "String":
|
||||
S7StringAttribute? attribute = propertyInfo?.GetCustomAttributes<S7StringAttribute>().SingleOrDefault();
|
||||
if (attribute == default(S7StringAttribute))
|
||||
throw new ArgumentException("Please add S7StringAttribute to the string field");
|
||||
|
||||
IncrementToEven(ref numBytes);
|
||||
numBytes += attribute.ReservedLengthInBytes;
|
||||
break;
|
||||
default:
|
||||
var propertyClass = Activator.CreateInstance(type);
|
||||
var propertyClass = Activator.CreateInstance(type) ??
|
||||
throw new ArgumentException($"Failed to create instance of type {type}.", nameof(type));
|
||||
numBytes = GetClassSize(propertyClass, numBytes, true);
|
||||
break;
|
||||
}
|
||||
@@ -75,6 +77,8 @@ namespace S7.Net.Types
|
||||
/// Gets the size of the class in bytes.
|
||||
/// </summary>
|
||||
/// <param name="instance">An instance of the class</param>
|
||||
/// <param name="numBytes">The offset of the current field.</param>
|
||||
/// <param name="isInnerProperty"><see langword="true" /> if this property belongs to a class being serialized as member of the class requested for serialization; otherwise, <see langword="false" />.</param>
|
||||
/// <returns>the number of bytes</returns>
|
||||
public static double GetClassSize(object instance, double numBytes = 0.0, bool isInnerProperty = false)
|
||||
{
|
||||
@@ -83,8 +87,10 @@ namespace S7.Net.Types
|
||||
{
|
||||
if (property.PropertyType.IsArray)
|
||||
{
|
||||
Type elementType = property.PropertyType.GetElementType();
|
||||
Array array = (Array)property.GetValue(instance, null);
|
||||
Type elementType = property.PropertyType.GetElementType()!;
|
||||
Array array = (Array?) property.GetValue(instance, null) ??
|
||||
throw new ArgumentException($"Property {property.Name} on {instance} must have a non-null value to get it's size.", nameof(instance));
|
||||
|
||||
if (array.Length <= 0)
|
||||
{
|
||||
throw new Exception("Cannot determine size of class, because an array is defined which has no fixed size greater than zero.");
|
||||
@@ -93,12 +99,12 @@ namespace S7.Net.Types
|
||||
IncrementToEven(ref numBytes);
|
||||
for (int i = 0; i < array.Length; i++)
|
||||
{
|
||||
numBytes = GetIncreasedNumberOfBytes(numBytes, elementType);
|
||||
numBytes = GetIncreasedNumberOfBytes(numBytes, elementType, property);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
numBytes = GetIncreasedNumberOfBytes(numBytes, property.PropertyType);
|
||||
numBytes = GetIncreasedNumberOfBytes(numBytes, property.PropertyType, property);
|
||||
}
|
||||
}
|
||||
if (false == isInnerProperty)
|
||||
@@ -111,7 +117,7 @@ namespace S7.Net.Types
|
||||
return numBytes;
|
||||
}
|
||||
|
||||
private static object? GetPropertyValue(Type propertyType, byte[] bytes, ref double numBytes)
|
||||
private static object? GetPropertyValue(Type propertyType, PropertyInfo? propertyInfo, byte[] bytes, ref double numBytes)
|
||||
{
|
||||
object? value = null;
|
||||
|
||||
@@ -133,50 +139,35 @@ namespace S7.Net.Types
|
||||
numBytes++;
|
||||
break;
|
||||
case "Int16":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
// hier auswerten
|
||||
ushort source = Word.FromBytes(bytes[(int)numBytes + 1], bytes[(int)numBytes]);
|
||||
value = source.ConvertToShort();
|
||||
numBytes += 2;
|
||||
break;
|
||||
case "UInt16":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
// hier auswerten
|
||||
value = Word.FromBytes(bytes[(int)numBytes + 1], bytes[(int)numBytes]);
|
||||
numBytes += 2;
|
||||
break;
|
||||
case "Int32":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
// hier auswerten
|
||||
uint sourceUInt = DWord.FromBytes(bytes[(int)numBytes + 3],
|
||||
bytes[(int)numBytes + 2],
|
||||
bytes[(int)numBytes + 1],
|
||||
bytes[(int)numBytes + 0]);
|
||||
IncrementToEven(ref numBytes);
|
||||
var wordBuffer = new byte[4];
|
||||
Array.Copy(bytes, (int)numBytes, wordBuffer, 0, wordBuffer.Length);
|
||||
uint sourceUInt = DWord.FromByteArray(wordBuffer);
|
||||
value = sourceUInt.ConvertToInt();
|
||||
numBytes += 4;
|
||||
break;
|
||||
case "UInt32":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
// hier auswerten
|
||||
value = DWord.FromBytes(
|
||||
bytes[(int)numBytes],
|
||||
bytes[(int)numBytes + 1],
|
||||
bytes[(int)numBytes + 2],
|
||||
bytes[(int)numBytes + 3]);
|
||||
IncrementToEven(ref numBytes);
|
||||
var wordBuffer2 = new byte[4];
|
||||
Array.Copy(bytes, (int)numBytes, wordBuffer2, 0, wordBuffer2.Length);
|
||||
value = DWord.FromByteArray(wordBuffer2);
|
||||
numBytes += 4;
|
||||
break;
|
||||
case "Single":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
// hier auswerten
|
||||
value = Real.FromByteArray(
|
||||
new byte[] {
|
||||
@@ -187,17 +178,35 @@ namespace S7.Net.Types
|
||||
numBytes += 4;
|
||||
break;
|
||||
case "Double":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
IncrementToEven(ref numBytes);
|
||||
var buffer = new byte[8];
|
||||
Array.Copy(bytes, (int)numBytes, buffer, 0, 8);
|
||||
// hier auswerten
|
||||
value = LReal.FromByteArray(buffer);
|
||||
numBytes += 8;
|
||||
break;
|
||||
case "String":
|
||||
S7StringAttribute? attribute = propertyInfo?.GetCustomAttributes<S7StringAttribute>().SingleOrDefault();
|
||||
if (attribute == default(S7StringAttribute))
|
||||
throw new ArgumentException("Please add S7StringAttribute to the string field");
|
||||
|
||||
IncrementToEven(ref numBytes);
|
||||
|
||||
// get the value
|
||||
var sData = new byte[attribute.ReservedLengthInBytes];
|
||||
Array.Copy(bytes, (int)numBytes, sData, 0, sData.Length);
|
||||
value = attribute.Type switch
|
||||
{
|
||||
S7StringType.S7String => S7String.FromByteArray(sData),
|
||||
S7StringType.S7WString => S7WString.FromByteArray(sData),
|
||||
_ => throw new ArgumentException("Please use a valid string type for the S7StringAttribute")
|
||||
};
|
||||
numBytes += sData.Length;
|
||||
break;
|
||||
default:
|
||||
var propClass = Activator.CreateInstance(propertyType);
|
||||
var propClass = Activator.CreateInstance(propertyType) ??
|
||||
throw new ArgumentException($"Failed to create instance of type {propertyType}.", nameof(propertyType));
|
||||
|
||||
numBytes = FromBytes(propClass, bytes, numBytes);
|
||||
value = propClass;
|
||||
break;
|
||||
@@ -211,6 +220,8 @@ namespace S7.Net.Types
|
||||
/// </summary>
|
||||
/// <param name="sourceClass">The object to fill in the given array of bytes</param>
|
||||
/// <param name="bytes">The array of bytes</param>
|
||||
/// <param name="numBytes">The offset for the current field.</param>
|
||||
/// <param name="isInnerClass"><see langword="true" /> if this class is the type of a member of the class to be serialized; otherwise, <see langword="false" />.</param>
|
||||
public static double FromBytes(object sourceClass, byte[] bytes, double numBytes = 0, bool isInnerClass = false)
|
||||
{
|
||||
if (bytes == null)
|
||||
@@ -221,13 +232,15 @@ namespace S7.Net.Types
|
||||
{
|
||||
if (property.PropertyType.IsArray)
|
||||
{
|
||||
Array array = (Array)property.GetValue(sourceClass, null);
|
||||
Array array = (Array?) property.GetValue(sourceClass, null) ??
|
||||
throw new ArgumentException($"Property {property.Name} on sourceClass must be an array instance.", nameof(sourceClass));
|
||||
|
||||
IncrementToEven(ref numBytes);
|
||||
Type elementType = property.PropertyType.GetElementType();
|
||||
Type elementType = property.PropertyType.GetElementType()!;
|
||||
for (int i = 0; i < array.Length && numBytes < bytes.Length; i++)
|
||||
{
|
||||
array.SetValue(
|
||||
GetPropertyValue(elementType, bytes, ref numBytes),
|
||||
GetPropertyValue(elementType, property, bytes, ref numBytes),
|
||||
i);
|
||||
}
|
||||
}
|
||||
@@ -235,7 +248,7 @@ namespace S7.Net.Types
|
||||
{
|
||||
property.SetValue(
|
||||
sourceClass,
|
||||
GetPropertyValue(property.PropertyType, bytes, ref numBytes),
|
||||
GetPropertyValue(property.PropertyType, property, bytes, ref numBytes),
|
||||
null);
|
||||
}
|
||||
}
|
||||
@@ -243,7 +256,7 @@ namespace S7.Net.Types
|
||||
return numBytes;
|
||||
}
|
||||
|
||||
private static double SetBytesFromProperty(object propertyValue, byte[] bytes, double numBytes)
|
||||
private static double SetBytesFromProperty(object propertyValue, PropertyInfo? propertyInfo, byte[] bytes, double numBytes)
|
||||
{
|
||||
int bytePos = 0;
|
||||
int bitPos = 0;
|
||||
@@ -285,6 +298,18 @@ namespace S7.Net.Types
|
||||
case "Double":
|
||||
bytes2 = LReal.ToByteArray((double)propertyValue);
|
||||
break;
|
||||
case "String":
|
||||
S7StringAttribute? attribute = propertyInfo?.GetCustomAttributes<S7StringAttribute>().SingleOrDefault();
|
||||
if (attribute == default(S7StringAttribute))
|
||||
throw new ArgumentException("Please add S7StringAttribute to the string field");
|
||||
|
||||
bytes2 = attribute.Type switch
|
||||
{
|
||||
S7StringType.S7String => S7String.ToByteArray((string)propertyValue, attribute.ReservedLength),
|
||||
S7StringType.S7WString => S7WString.ToByteArray((string)propertyValue, attribute.ReservedLength),
|
||||
_ => throw new ArgumentException("Please use a valid string type for the S7StringAttribute")
|
||||
};
|
||||
break;
|
||||
default:
|
||||
numBytes = ToBytes(propertyValue, bytes, numBytes);
|
||||
break;
|
||||
@@ -306,26 +331,30 @@ namespace S7.Net.Types
|
||||
/// <summary>
|
||||
/// Creates a byte array depending on the struct type.
|
||||
/// </summary>
|
||||
/// <param name="sourceClass">The struct object</param>
|
||||
/// <param name="sourceClass">The struct object.</param>
|
||||
/// <param name="bytes">The target byte array.</param>
|
||||
/// <param name="numBytes">The offset for the current field.</param>
|
||||
/// <returns>A byte array or null if fails.</returns>
|
||||
public static double ToBytes(object sourceClass, byte[] bytes, double numBytes = 0.0)
|
||||
{
|
||||
var properties = GetAccessableProperties(sourceClass.GetType());
|
||||
foreach (var property in properties)
|
||||
{
|
||||
var value = property.GetValue(sourceClass, null) ??
|
||||
throw new ArgumentException($"Property {property.Name} on sourceClass can't be null.", nameof(sourceClass));
|
||||
|
||||
if (property.PropertyType.IsArray)
|
||||
{
|
||||
Array array = (Array)property.GetValue(sourceClass, null);
|
||||
Array array = (Array) value;
|
||||
IncrementToEven(ref numBytes);
|
||||
Type elementType = property.PropertyType.GetElementType();
|
||||
for (int i = 0; i < array.Length && numBytes < bytes.Length; i++)
|
||||
{
|
||||
numBytes = SetBytesFromProperty(array.GetValue(i), bytes, numBytes);
|
||||
numBytes = SetBytesFromProperty(array.GetValue(i)!, property, bytes, numBytes);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
numBytes = SetBytesFromProperty(property.GetValue(sourceClass, null), bytes, numBytes);
|
||||
numBytes = SetBytesFromProperty(value, property, bytes, numBytes);
|
||||
}
|
||||
}
|
||||
return numBytes;
|
||||
|
||||
82
S7.Net/Types/Date.cs
Normal file
82
S7.Net/Types/Date.cs
Normal file
@@ -0,0 +1,82 @@
|
||||
using System;
|
||||
using S7.Net.Helper;
|
||||
|
||||
namespace S7.Net.Types
|
||||
{
|
||||
/// <summary>
|
||||
/// Contains the conversion methods to convert Words from S7 plc to C#.
|
||||
/// </summary>
|
||||
public static class Date
|
||||
{
|
||||
/// <summary>
|
||||
/// Minimum allowed date for the IEC date type
|
||||
/// </summary>
|
||||
public static System.DateTime IecMinDate { get; } = new(year: 1990, month: 01, day: 01);
|
||||
|
||||
/// <summary>
|
||||
/// Maximum allowed date for the IEC date type
|
||||
/// <remarks>
|
||||
/// Although the spec allows only a max date of 31-12-2168, the PLC IEC date goes up to 06-06-2169 (which is the actual
|
||||
/// WORD max value - 65535)
|
||||
/// </remarks>
|
||||
/// </summary>
|
||||
public static System.DateTime IecMaxDate { get; } = new(year: 2169, month: 06, day: 06);
|
||||
|
||||
private static readonly ushort MaxNumberOfDays = (ushort)(IecMaxDate - IecMinDate).TotalDays;
|
||||
|
||||
/// <summary>
|
||||
/// Converts a word (2 bytes) to IEC date (<see cref="System.DateTime"/>)
|
||||
/// </summary>
|
||||
public static System.DateTime FromByteArray(byte[] bytes)
|
||||
{
|
||||
if (bytes.Length != 2)
|
||||
{
|
||||
throw new ArgumentException("Wrong number of bytes. Bytes array must contain 2 bytes.");
|
||||
}
|
||||
|
||||
var daysSinceDateStart = Word.FromByteArray(bytes);
|
||||
if (daysSinceDateStart > MaxNumberOfDays)
|
||||
{
|
||||
throw new ArgumentException($"Read number exceeded the number of maximum days in the IEC date (read: {daysSinceDateStart}, max: {MaxNumberOfDays})",
|
||||
nameof(bytes));
|
||||
}
|
||||
|
||||
return IecMinDate.AddDays(daysSinceDateStart);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a <see cref="System.DateTime"/> to word (2 bytes)
|
||||
/// </summary>
|
||||
public static byte[] ToByteArray(System.DateTime dateTime) => Word.ToByteArray(dateTime.GetDaysSinceIecDateStart());
|
||||
|
||||
/// <summary>
|
||||
/// Converts an array of <see cref="System.DateTime"/>s to an array of bytes
|
||||
/// </summary>
|
||||
public static byte[] ToByteArray(System.DateTime[] value)
|
||||
{
|
||||
var arr = new ByteArray();
|
||||
foreach (var date in value)
|
||||
arr.Add(ToByteArray(date));
|
||||
return arr.Array;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts an array of bytes to an array of <see cref="System.DateTime"/>s
|
||||
/// </summary>
|
||||
public static System.DateTime[] ToArray(byte[] bytes)
|
||||
{
|
||||
var values = new System.DateTime[bytes.Length / sizeof(ushort)];
|
||||
|
||||
for (int i = 0; i < values.Length; i++)
|
||||
{
|
||||
values[i] = FromByteArray(
|
||||
new[]
|
||||
{
|
||||
bytes[i], bytes[i + 1]
|
||||
});
|
||||
}
|
||||
|
||||
return values;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -141,7 +141,7 @@ namespace S7.Net.Types
|
||||
/// Converts an array of <see cref="T:System.DateTime"/> values to a byte array.
|
||||
/// </summary>
|
||||
/// <param name="dateTimes">The DateTime values to convert.</param>
|
||||
/// <returns>A byte array containing the S7 date time representations of <paramref name="dateTime"/>.</returns>
|
||||
/// <returns>A byte array containing the S7 date time representations of <paramref name="dateTimes"/>.</returns>
|
||||
/// <exception cref="ArgumentOutOfRangeException">Thrown when any value of
|
||||
/// <paramref name="dateTimes"/> is before <see cref="P:SpecMinimumDateTime"/>
|
||||
/// or after <see cref="P:SpecMaximumDateTime"/>.</exception>
|
||||
|
||||
@@ -8,17 +8,17 @@ namespace S7.Net.Types
|
||||
/// An S7 String has a preceeding 2 byte header containing its capacity and length
|
||||
/// </summary>
|
||||
public static class S7String
|
||||
{
|
||||
private static Encoding stringEncoding = Encoding.ASCII;
|
||||
|
||||
{
|
||||
private static Encoding stringEncoding = Encoding.ASCII;
|
||||
|
||||
/// <summary>
|
||||
/// The Encoding used when serializing and deserializing S7String (Encoding.ASCII by default)
|
||||
/// </summary>
|
||||
/// <exception cref="ArgumentNullException">StringEncoding must not be null</exception>
|
||||
public static Encoding StringEncoding
|
||||
{
|
||||
get => stringEncoding;
|
||||
set => stringEncoding = value ?? throw new ArgumentNullException(nameof(StringEncoding));
|
||||
/// <exception cref="ArgumentNullException">StringEncoding must not be null</exception>
|
||||
public static Encoding StringEncoding
|
||||
{
|
||||
get => stringEncoding;
|
||||
set => stringEncoding = value ?? throw new ArgumentNullException(nameof(StringEncoding));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -58,7 +58,7 @@ namespace S7.Net.Types
|
||||
/// <param name="value">The string to convert to byte array.</param>
|
||||
/// <param name="reservedLength">The length (in characters) allocated in PLC for the string.</param>
|
||||
/// <returns>A <see cref="T:byte[]" /> containing the string header and string value with a maximum length of <paramref name="reservedLength"/> + 2.</returns>
|
||||
public static byte[] ToByteArray(string value, int reservedLength)
|
||||
public static byte[] ToByteArray(string? value, int reservedLength)
|
||||
{
|
||||
if (value is null)
|
||||
{
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
namespace S7.Net.Types
|
||||
{
|
||||
[AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
|
||||
[AttributeUsage(AttributeTargets.Field | AttributeTargets.Property, AllowMultiple = false)]
|
||||
public sealed class S7StringAttribute : Attribute
|
||||
{
|
||||
private readonly S7StringType type;
|
||||
|
||||
@@ -48,7 +48,7 @@ namespace S7.Net.Types
|
||||
/// <param name="value">The string to convert to byte array.</param>
|
||||
/// <param name="reservedLength">The length (in characters) allocated in PLC for the string.</param>
|
||||
/// <returns>A <see cref="T:byte[]" /> containing the string header and string value with a maximum length of <paramref name="reservedLength"/> + 4.</returns>
|
||||
public static byte[] ToByteArray(string value, int reservedLength)
|
||||
public static byte[] ToByteArray(string? value, int reservedLength)
|
||||
{
|
||||
if (value is null)
|
||||
{
|
||||
|
||||
@@ -12,13 +12,15 @@
|
||||
/// <param name="reservedLength">The amount of bytes reserved for the <paramref name="value"/> in the PLC.</param>
|
||||
public static byte[] ToByteArray(string value, int reservedLength)
|
||||
{
|
||||
var length = value?.Length;
|
||||
if (length > reservedLength) length = reservedLength;
|
||||
var bytes = new byte[reservedLength];
|
||||
if (value == null) return bytes;
|
||||
|
||||
if (length == null || length == 0) return bytes;
|
||||
var length = value.Length;
|
||||
if (length == 0) return bytes;
|
||||
|
||||
System.Text.Encoding.ASCII.GetBytes(value, 0, length.Value, bytes, 0);
|
||||
if (length > reservedLength) length = reservedLength;
|
||||
|
||||
System.Text.Encoding.ASCII.GetBytes(value, 0, length, bytes, 0);
|
||||
|
||||
return bytes;
|
||||
}
|
||||
|
||||
@@ -45,6 +45,7 @@ namespace S7.Net.Types
|
||||
break;
|
||||
case "Int32":
|
||||
case "UInt32":
|
||||
case "TimeSpan":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
@@ -98,8 +99,8 @@ namespace S7.Net.Types
|
||||
int bytePos = 0;
|
||||
int bitPos = 0;
|
||||
double numBytes = 0.0;
|
||||
object structValue = Activator.CreateInstance(structType);
|
||||
|
||||
object structValue = Activator.CreateInstance(structType) ??
|
||||
throw new ArgumentException($"Failed to create an instance of the type {structType}.", nameof(structType));
|
||||
|
||||
var infos = structValue.GetType()
|
||||
#if NETSTANDARD1_3
|
||||
@@ -215,6 +216,21 @@ namespace S7.Net.Types
|
||||
|
||||
numBytes += sData.Length;
|
||||
break;
|
||||
case "TimeSpan":
|
||||
numBytes = Math.Ceiling(numBytes);
|
||||
if ((numBytes / 2 - Math.Floor(numBytes / 2.0)) > 0)
|
||||
numBytes++;
|
||||
|
||||
// get the value
|
||||
info.SetValue(structValue, TimeSpan.FromByteArray(new[]
|
||||
{
|
||||
bytes[(int)numBytes + 0],
|
||||
bytes[(int)numBytes + 1],
|
||||
bytes[(int)numBytes + 2],
|
||||
bytes[(int)numBytes + 3]
|
||||
}));
|
||||
numBytes += 4;
|
||||
break;
|
||||
default:
|
||||
var buffer = new byte[GetStructSize(info.FieldType)];
|
||||
if (buffer.Length == 0)
|
||||
@@ -254,6 +270,14 @@ namespace S7.Net.Types
|
||||
|
||||
foreach (var info in infos)
|
||||
{
|
||||
static TValue GetValueOrThrow<TValue>(FieldInfo fi, object structValue) where TValue : struct
|
||||
{
|
||||
var value = fi.GetValue(structValue) as TValue? ??
|
||||
throw new ArgumentException($"Failed to convert value of field {fi.Name} of {structValue} to type {typeof(TValue)}");
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
bytes2 = null;
|
||||
switch (info.FieldType.Name)
|
||||
{
|
||||
@@ -261,7 +285,7 @@ namespace S7.Net.Types
|
||||
// get the value
|
||||
bytePos = (int)Math.Floor(numBytes);
|
||||
bitPos = (int)((numBytes - (double)bytePos) / 0.125);
|
||||
if ((bool)info.GetValue(structValue))
|
||||
if (GetValueOrThrow<bool>(info, structValue))
|
||||
bytes[bytePos] |= (byte)Math.Pow(2, bitPos); // is true
|
||||
else
|
||||
bytes[bytePos] &= (byte)(~(byte)Math.Pow(2, bitPos)); // is false
|
||||
@@ -270,26 +294,26 @@ namespace S7.Net.Types
|
||||
case "Byte":
|
||||
numBytes = (int)Math.Ceiling(numBytes);
|
||||
bytePos = (int)numBytes;
|
||||
bytes[bytePos] = (byte)info.GetValue(structValue);
|
||||
bytes[bytePos] = GetValueOrThrow<byte>(info, structValue);
|
||||
numBytes++;
|
||||
break;
|
||||
case "Int16":
|
||||
bytes2 = Int.ToByteArray((Int16)info.GetValue(structValue));
|
||||
bytes2 = Int.ToByteArray(GetValueOrThrow<short>(info, structValue));
|
||||
break;
|
||||
case "UInt16":
|
||||
bytes2 = Word.ToByteArray((UInt16)info.GetValue(structValue));
|
||||
bytes2 = Word.ToByteArray(GetValueOrThrow<ushort>(info, structValue));
|
||||
break;
|
||||
case "Int32":
|
||||
bytes2 = DInt.ToByteArray((Int32)info.GetValue(structValue));
|
||||
bytes2 = DInt.ToByteArray(GetValueOrThrow<int>(info, structValue));
|
||||
break;
|
||||
case "UInt32":
|
||||
bytes2 = DWord.ToByteArray((UInt32)info.GetValue(structValue));
|
||||
bytes2 = DWord.ToByteArray(GetValueOrThrow<uint>(info, structValue));
|
||||
break;
|
||||
case "Single":
|
||||
bytes2 = Real.ToByteArray((float)info.GetValue(structValue));
|
||||
bytes2 = Real.ToByteArray(GetValueOrThrow<float>(info, structValue));
|
||||
break;
|
||||
case "Double":
|
||||
bytes2 = LReal.ToByteArray((double)info.GetValue(structValue));
|
||||
bytes2 = LReal.ToByteArray(GetValueOrThrow<double>(info, structValue));
|
||||
break;
|
||||
case "String":
|
||||
S7StringAttribute? attribute = info.GetCustomAttributes<S7StringAttribute>().SingleOrDefault();
|
||||
@@ -298,11 +322,14 @@ namespace S7.Net.Types
|
||||
|
||||
bytes2 = attribute.Type switch
|
||||
{
|
||||
S7StringType.S7String => S7String.ToByteArray((string)info.GetValue(structValue), attribute.ReservedLength),
|
||||
S7StringType.S7WString => S7WString.ToByteArray((string)info.GetValue(structValue), attribute.ReservedLength),
|
||||
S7StringType.S7String => S7String.ToByteArray((string?)info.GetValue(structValue), attribute.ReservedLength),
|
||||
S7StringType.S7WString => S7WString.ToByteArray((string?)info.GetValue(structValue), attribute.ReservedLength),
|
||||
_ => throw new ArgumentException("Please use a valid string type for the S7StringAttribute")
|
||||
};
|
||||
break;
|
||||
case "TimeSpan":
|
||||
bytes2 = TimeSpan.ToByteArray((System.TimeSpan)info.GetValue(structValue));
|
||||
break;
|
||||
}
|
||||
if (bytes2 != null)
|
||||
{
|
||||
|
||||
97
S7.Net/Types/TimeSpan.cs
Normal file
97
S7.Net/Types/TimeSpan.cs
Normal file
@@ -0,0 +1,97 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace S7.Net.Types
|
||||
{
|
||||
/// <summary>
|
||||
/// Contains the methods to convert between <see cref="T:System.TimeSpan"/> and S7 representation of TIME values.
|
||||
/// </summary>
|
||||
public static class TimeSpan
|
||||
{
|
||||
/// <summary>
|
||||
/// The minimum <see cref="T:System.TimeSpan"/> value supported by the specification.
|
||||
/// </summary>
|
||||
public static readonly System.TimeSpan SpecMinimumTimeSpan = System.TimeSpan.FromMilliseconds(int.MinValue);
|
||||
|
||||
/// <summary>
|
||||
/// The maximum <see cref="T:System.TimeSpan"/> value supported by the specification.
|
||||
/// </summary>
|
||||
public static readonly System.TimeSpan SpecMaximumTimeSpan = System.TimeSpan.FromMilliseconds(int.MaxValue);
|
||||
|
||||
/// <summary>
|
||||
/// Parses a <see cref="T:System.TimeSpan"/> value from bytes.
|
||||
/// </summary>
|
||||
/// <param name="bytes">Input bytes read from PLC.</param>
|
||||
/// <returns>A <see cref="T:System.TimeSpan"/> object representing the value read from PLC.</returns>
|
||||
/// <exception cref="ArgumentOutOfRangeException">Thrown when the length of
|
||||
/// <paramref name="bytes"/> is not 4 or any value in <paramref name="bytes"/>
|
||||
/// is outside the valid range of values.</exception>
|
||||
public static System.TimeSpan FromByteArray(byte[] bytes)
|
||||
{
|
||||
var milliseconds = DInt.FromByteArray(bytes);
|
||||
return System.TimeSpan.FromMilliseconds(milliseconds);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Parses an array of <see cref="T:System.TimeSpan"/> values from bytes.
|
||||
/// </summary>
|
||||
/// <param name="bytes">Input bytes read from PLC.</param>
|
||||
/// <returns>An array of <see cref="T:System.TimeSpan"/> objects representing the values read from PLC.</returns>
|
||||
/// <exception cref="ArgumentOutOfRangeException">Thrown when the length of
|
||||
/// <paramref name="bytes"/> is not a multiple of 4 or any value in
|
||||
/// <paramref name="bytes"/> is outside the valid range of values.</exception>
|
||||
public static System.TimeSpan[] ToArray(byte[] bytes)
|
||||
{
|
||||
const int singleTimeSpanLength = 4;
|
||||
|
||||
if (bytes.Length % singleTimeSpanLength != 0)
|
||||
throw new ArgumentOutOfRangeException(nameof(bytes), bytes.Length,
|
||||
$"Parsing an array of {nameof(System.TimeSpan)} requires a multiple of {singleTimeSpanLength} bytes of input data, input data is '{bytes.Length}' long.");
|
||||
|
||||
var result = new System.TimeSpan[bytes.Length / singleTimeSpanLength];
|
||||
|
||||
var milliseconds = DInt.ToArray(bytes);
|
||||
for (var i = 0; i < milliseconds.Length; i++)
|
||||
result[i] = System.TimeSpan.FromMilliseconds(milliseconds[i]);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a <see cref="T:System.TimeSpan"/> value to a byte array.
|
||||
/// </summary>
|
||||
/// <param name="timeSpan">The TimeSpan value to convert.</param>
|
||||
/// <returns>A byte array containing the S7 date time representation of <paramref name="timeSpan"/>.</returns>
|
||||
/// <exception cref="ArgumentOutOfRangeException">Thrown when the value of
|
||||
/// <paramref name="timeSpan"/> is before <see cref="P:SpecMinimumTimeSpan"/>
|
||||
/// or after <see cref="P:SpecMaximumTimeSpan"/>.</exception>
|
||||
public static byte[] ToByteArray(System.TimeSpan timeSpan)
|
||||
{
|
||||
if (timeSpan < SpecMinimumTimeSpan)
|
||||
throw new ArgumentOutOfRangeException(nameof(timeSpan), timeSpan,
|
||||
$"Time span '{timeSpan}' is before the minimum '{SpecMinimumTimeSpan}' supported in S7 time representation.");
|
||||
|
||||
if (timeSpan > SpecMaximumTimeSpan)
|
||||
throw new ArgumentOutOfRangeException(nameof(timeSpan), timeSpan,
|
||||
$"Time span '{timeSpan}' is after the maximum '{SpecMaximumTimeSpan}' supported in S7 time representation.");
|
||||
|
||||
return DInt.ToByteArray(Convert.ToInt32(timeSpan.TotalMilliseconds));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts an array of <see cref="T:System.TimeSpan"/> values to a byte array.
|
||||
/// </summary>
|
||||
/// <param name="timeSpans">The TimeSpan values to convert.</param>
|
||||
/// <returns>A byte array containing the S7 date time representations of <paramref name="timeSpans"/>.</returns>
|
||||
/// <exception cref="ArgumentOutOfRangeException">Thrown when any value of
|
||||
/// <paramref name="timeSpans"/> is before <see cref="P:SpecMinimumTimeSpan"/>
|
||||
/// or after <see cref="P:SpecMaximumTimeSpan"/>.</exception>
|
||||
public static byte[] ToByteArray(System.TimeSpan[] timeSpans)
|
||||
{
|
||||
var bytes = new List<byte>(timeSpans.Length * 4);
|
||||
foreach (var timeSpan in timeSpans) bytes.AddRange(ToByteArray(timeSpan));
|
||||
|
||||
return bytes.ToArray();
|
||||
}
|
||||
}
|
||||
}
|
||||
13
appveyor.yml
13
appveyor.yml
@@ -1,13 +0,0 @@
|
||||
image: Visual Studio 2019
|
||||
configuration: Release
|
||||
install:
|
||||
- choco install gitversion.portable -y
|
||||
before_build:
|
||||
- cmd: gitversion /l console /output buildserver
|
||||
- dotnet restore
|
||||
build_script:
|
||||
msbuild /nologo /v:m /p:AssemblyVersion=%GitVersion_AssemblySemVer% /p:FileVersion=%GitVersion_MajorMinorPatch% /p:InformationalVersion=%GitVersion_InformationalVersion% /p:Configuration=%CONFIGURATION% S7.sln
|
||||
after_build:
|
||||
- dotnet pack S7.Net -c %CONFIGURATION% /p:Version=%GitVersion_NuGetVersion% --no-build -o artifacts
|
||||
artifacts:
|
||||
- path: artifacts\*.*
|
||||
Reference in New Issue
Block a user