Class: PlcAccess::Protocol::Mitsubishi::McProtocol
- Inherits:
-
Protocol
- Object
- Protocol
- PlcAccess::Protocol::Mitsubishi::McProtocol
show all
- Defined in:
- lib/plc_access/protocol/mitsubishi/mc_protocol.rb
Constant Summary
Constants inherited
from Protocol
Protocol::TIMEOUT
Instance Attribute Summary
Attributes inherited from Protocol
#host, #log_level, #port
Instance Method Summary
collapse
Methods inherited from Protocol
#[], #[]=, #destination_ipv4, #get_bit_from_device, #get_from_devices, #get_word_from_device, #self_ipv4, #set_bit_to_device, #set_to_devices, #set_word_to_device
Constructor Details
#initialize(options = {}) ⇒ McProtocol
Returns a new instance of McProtocol.
30
31
32
33
34
35
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 30
def initialize(options = {})
super
@socket = nil
@host = options[:host] || '192.168.0.10'
@port = options[:port] || 5010
end
|
Instance Method Details
#available_bits_range(_device = nil) ⇒ Object
195
196
197
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 195
def available_bits_range(_device = nil)
1..(960 * 16)
end
|
#available_words_range(_device = nil) ⇒ Object
199
200
201
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 199
def available_words_range(_device = nil)
1..960
end
|
#close ⇒ Object
47
48
49
50
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 47
def close
@socket&.close
@socket = nil
end
|
#device_by_name(name) ⇒ Object
163
164
165
166
167
168
169
170
171
172
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 163
def device_by_name(name)
case name
when String
d = QDevice.new name
d.valid? ? d : nil
else
name
end
end
|
#get_bits_from_device(count, device) ⇒ Object
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
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 52
def get_bits_from_device(count, device)
unless available_bits_range.include? count
raise ArgumentError,
"A count #{count} must be between #{available_bits_range.first} and #{available_bits_range.last} for #{__method__}"
end
device = device_by_name device
packet = make_packet(body_for_get_bits_from_device(count, device))
@logger.debug("> #{dump_packet packet}")
open
@socket.write(packet.pack('C*'))
@socket.flush
res = receive
end_code = res[9, 2].pack('C*').unpack1('v')
unless end_code.zero?
error = res[11, 2].pack('C*').unpack1('v')
raise "return end code 0x#{end_code.to_s(16)} error code 0x#{error.to_s(16)} for get_bits_from_device(#{count}, #{device.name})"
end
bits = []
count.times do |i|
v = res[11 + i / 2]
bits << if i.even?
((v >> 4) != 0)
else
((v & 0xf) != 0)
end
end
@logger.debug("get #{device.name} => #{bits}")
bits
end
|
#get_words_from_device(count, device) ⇒ Object
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
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 110
def get_words_from_device(count, device)
unless available_bits_range.include? count
raise ArgumentError,
"A count #{count} must be between #{available_words_range.first} and #{available_words_range.last} for #{__method__}"
end
device = device_by_name device
packet = make_packet(body_for_get_words_from_device(count, device))
@logger.debug("> #{dump_packet packet}")
open
@socket.write(packet.pack('C*'))
@socket.flush
res = receive
end_code = res[9, 2].pack('C*').unpack1('v')
unless end_code.zero?
error = res[11, 2].pack('C*').unpack1('v')
raise "return end code 0x#{end_code.to_s(16)} error code 0x#{error.to_s(16)} for get_words_from_device(#{count}, #{device.name})"
end
words = []
res[11, 2 * count].each_slice(2) do |pair|
words << pair.pack('C*').unpack1('v')
end
@logger.debug("get from: #{device.name} => #{words}")
words
end
|
#open ⇒ Object
37
38
39
40
41
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 37
def open
open!
rescue StandardError
nil
end
|
#open! ⇒ Object
43
44
45
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 43
def open!
@socket ||= TCPSocket.open(@host, @port)
end
|
#receive ⇒ Object
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 174
def receive
res = []
len = 0
begin
Timeout.timeout(TIMEOUT) do
loop do
c = @socket.read(1)
next if c.nil? || c == ''
res << c.bytes.first
len = res[7, 2].pack('C*').unpack1('v*') if res.length >= 9
break if len + 9 == res.length
end
end
rescue Timeout::Error
puts '*** ERROR: TIME OUT ***'
end
@logger.debug("< #{dump_packet res}")
res
end
|
#set_bits_to_device(bits, device) ⇒ Object
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 87
def set_bits_to_device(bits, device)
unless available_bits_range.include? bits.size
raise ArgumentError,
"A count #{count} must be between #{available_bits_range.first} and #{available_bits_range.last} for #{__method__}"
end
device = device_by_name device
packet = make_packet(body_for_set_bits_to_device(bits, device))
@logger.debug("> #{dump_packet packet}")
open
@socket.write(packet.pack('C*'))
@socket.flush
res = receive
@logger.debug("set #{bits} to:#{device.name}")
end_code = res[9, 2].pack('C*').unpack1('v')
unless end_code.zero?
error = res[11, 2].pack('C*').unpack1('v')
raise "return end code 0x#{end_code.to_s(16)} error code 0x#{error.to_s(16)} for set_bits_to_device(#{bits}, #{device.name})"
end
end
|
#set_words_to_device(words, device) ⇒ Object
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
|
# File 'lib/plc_access/protocol/mitsubishi/mc_protocol.rb', line 140
def set_words_to_device(words, device)
unless available_bits_range.include? words.size
raise ArgumentError,
"A count of words #{words.size} must be between #{available_words_range.first} and #{available_words_range.last} for #{__method__}"
end
device = device_by_name device
packet = make_packet(body_for_set_words_to_device(words, device))
@logger.debug("> #{dump_packet packet}")
open
@socket.write(packet.pack('C*'))
@socket.flush
res = receive
@logger.debug("set #{words} to: #{device.name}")
end_code = res[9, 2].pack('C*').unpack1('v')
unless end_code.zero?
error = res[11, 2].pack('C*').unpack1('v')
raise "return end code 0x#{end_code.to_s(16)} error code 0x#{error.to_s(16)} for set_words_to_device(#{words}, #{device.name})"
end
end
|