forked from tbrandon/mbserver
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.go
198 lines (161 loc) · 4.52 KB
/
server.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
// Package mbserver implments a Modbus server (slave).
package mbserver
import (
"io"
"log/slog"
"net"
"sync"
"github.com/goburrow/serial"
)
// Server is a Modbus slave with allocated memory for discrete inputs, coils, etc.
type Server struct {
// Debug enables more verbose messaging.
Debug bool
listeners []net.Listener
ports []serial.Port
wg sync.WaitGroup
closeSignalChan chan struct{}
requestChan chan *Request
l *slog.Logger
function [256]function
DiscreteInputs []byte
Coils []byte
HoldingRegisters []uint16
InputRegisters []uint16
}
// Request contains the connection and Modbus frame.
type Request struct {
conn io.ReadWriteCloser
frame Framer
}
// OptionFunc is a function type used to configure options for the Server.
type OptionFunc func(s *Server)
// WithLogger sets the logger for the Server.
// Parameter l is a pointer to an *slog.Logger instance to be used for logging.
func WithLogger(l *slog.Logger) OptionFunc {
return func(s *Server) {
s.l = l
}
}
// WithDiscreteInputs sets the DiscreteInputs data for the Server.
// Parameter data is a byte slice containing the discrete input data to set.
func WithDiscreteInputs(data []byte) OptionFunc {
return func(s *Server) {
s.DiscreteInputs = data
}
}
// WithCoils sets the Coils data for the Server.
// Parameter data is a byte slice containing the coil data to set.
func WithCoils(data []byte) OptionFunc {
return func(s *Server) {
s.Coils = data
}
}
// WithHoldingRegisters sets the HoldingRegisters data for the Server.
// Parameter data is a byte slice containing the holding register data to set.
func WithHoldingRegisters(data []uint16) OptionFunc {
return func(s *Server) {
s.HoldingRegisters = data
}
}
// WithInputRegisters sets the InputRegisters data for the Server.
// Parameter data is a byte slice containing the input register data to set.
func WithInputRegisters(data []uint16) OptionFunc {
return func(s *Server) {
s.InputRegisters = data
}
}
// WithRegisterFunction registers a custom function handler for a specific function code.
// Parameter funcCode is the function code, and function is the custom handler for that code.
func WithRegisterFunction(funcCode uint8, function function) OptionFunc {
return func(s *Server) {
s.registerFunctionHandler(funcCode, function)
}
}
// NewServer creates a new Modbus server (slave).
func NewServer(opts ...OptionFunc) *Server {
s := &Server{}
// Add default functions.
s.function[1] = ReadCoils
s.function[2] = ReadDiscreteInputs
s.function[3] = ReadHoldingRegisters
s.function[4] = ReadInputRegisters
s.function[5] = WriteSingleCoil
s.function[6] = WriteHoldingRegister
s.function[15] = WriteMultipleCoils
s.function[16] = WriteHoldingRegisters
for _, opt := range opts {
opt(s)
}
// default slog.Logger
if s.l == nil {
s.l = slog.Default()
}
// Allocate Modbus memory maps.
if s.DiscreteInputs == nil {
s.DiscreteInputs = make([]byte, 65536)
}
if s.Coils == nil {
s.Coils = make([]byte, 65536)
}
if s.HoldingRegisters == nil {
s.HoldingRegisters = make([]uint16, 65536)
}
if s.InputRegisters == nil {
s.InputRegisters = make([]uint16, 65536)
}
s.requestChan = make(chan *Request)
s.closeSignalChan = make(chan struct{})
return s
}
// registerFunctionHandler override the default behavior for a given Modbus function.
func (s *Server) registerFunctionHandler(funcCode uint8, function function) {
s.function[funcCode] = function
}
func (s *Server) handle(request *Request) Framer {
var exception *Exception
var data []byte
response := request.frame.Copy()
funcCode := request.frame.GetFunction()
if s.function[funcCode] != nil {
data, exception = s.function[funcCode](s, request.frame)
response.SetData(data)
} else {
exception = &IllegalFunction
}
if *exception == Success {
response.SetException(exception)
}
return response
}
// All requests are handled synchronously to prevent modbus memory corruption.
func (s *Server) handler() {
for {
select {
case <-s.closeSignalChan:
return
case request := <-s.requestChan:
response := s.handle(request)
request.conn.Write(response.Bytes())
}
}
}
// Serve start the service
func (s *Server) Serve() {
for _, listener := range s.listeners {
go s.accept(listener)
}
for _, port := range s.ports {
s.wg.Add(1)
go func() {
defer s.wg.Done()
s.acceptSerialRequests(port)
}()
}
s.handler()
}
// Shutdown stops listening to TCP/IP ports and closes serial ports.
func (s *Server) Shutdown() {
close(s.closeSignalChan)
s.wg.Wait()
}