line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Biblio::RFID::Reader::3M810; |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
=head1 NAME |
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
Biblio::RFID::Reader::3M810 - support for 3M 810 RFID reader |
6
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=head1 DESCRIPTION |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
This module uses L over USB/serial adapter |
10
|
|
|
|
|
|
|
with 3M 810 RFID reader, often used in library applications. |
11
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
This is most mature implementation which supports full API defined |
13
|
|
|
|
|
|
|
in L. This include scanning for all tags in reader |
14
|
|
|
|
|
|
|
range, reading and writing of data, and AFI security manipulation. |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
This implementation is developed using Portmon on Windows to capture serial traffic |
17
|
|
|
|
|
|
|
L |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
Checksum for this reader is developed using help from C |
20
|
|
|
|
|
|
|
L |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
More inforation about process of reverse engeeniring protocol with |
23
|
|
|
|
|
|
|
this reader is available at L |
24
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
=cut |
26
|
|
|
|
|
|
|
|
27
|
3
|
|
|
3
|
|
67753
|
use warnings; |
|
3
|
|
|
|
|
18
|
|
|
3
|
|
|
|
|
111
|
|
28
|
3
|
|
|
3
|
|
17
|
use strict; |
|
3
|
|
|
|
|
7
|
|
|
3
|
|
|
|
|
105
|
|
29
|
|
|
|
|
|
|
|
30
|
3
|
|
|
3
|
|
16
|
use base 'Biblio::RFID::Reader::Serial'; |
|
3
|
|
|
|
|
7
|
|
|
3
|
|
|
|
|
1011
|
|
31
|
3
|
|
|
3
|
|
683
|
use Biblio::RFID; |
|
3
|
|
|
|
|
7
|
|
|
3
|
|
|
|
|
382
|
|
32
|
|
|
|
|
|
|
|
33
|
3
|
|
|
3
|
|
17
|
use Data::Dump qw(dump); |
|
3
|
|
|
|
|
5
|
|
|
3
|
|
|
|
|
124
|
|
34
|
3
|
|
|
3
|
|
18
|
use Carp qw(confess); |
|
3
|
|
|
|
|
15
|
|
|
3
|
|
|
|
|
134
|
|
35
|
3
|
|
|
3
|
|
1966
|
use Time::HiRes; |
|
3
|
|
|
|
|
4610
|
|
|
3
|
|
|
|
|
21
|
|
36
|
3
|
|
|
3
|
|
1759
|
use Digest::CRC; |
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
sub serial_settings {{ |
39
|
|
|
|
|
|
|
baudrate => "19200", |
40
|
|
|
|
|
|
|
databits => "8", |
41
|
|
|
|
|
|
|
parity => "none", |
42
|
|
|
|
|
|
|
stopbits => "1", |
43
|
|
|
|
|
|
|
handshake => "none", |
44
|
|
|
|
|
|
|
}} |
45
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
sub assert; |
47
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
my $port; |
49
|
|
|
|
|
|
|
sub init { |
50
|
|
|
|
|
|
|
my $self = shift; |
51
|
|
|
|
|
|
|
$port = $self->port; |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
# disable timeouts |
54
|
|
|
|
|
|
|
$port->read_char_time(0); |
55
|
|
|
|
|
|
|
$port->read_const_time(0); |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
# drain on startup |
58
|
|
|
|
|
|
|
my ( $count, $str ) = $port->read(3); |
59
|
|
|
|
|
|
|
if ( $count ) { |
60
|
|
|
|
|
|
|
my $data = $port->read( ord(substr($str,2,1)) ); |
61
|
|
|
|
|
|
|
warn "drain ",as_hex( $str, $data ),"\n"; |
62
|
|
|
|
|
|
|
} |
63
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
$port->read_char_time(100); # 0.1 s char timeout |
65
|
|
|
|
|
|
|
$port->read_const_time(500); # 0.5 s read timeout |
66
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
$port->write( hex2bytes( 'D5 00 05 04 00 11 8C66' ) ); |
68
|
|
|
|
|
|
|
# hw-version expect: 'D5 00 09 04 00 11 0A 05 00 02 7250' |
69
|
|
|
|
|
|
|
my $data = $port->read( 12 ); |
70
|
|
|
|
|
|
|
return unless $data; |
71
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
warn "# probe response: ",as_hex($data); |
73
|
|
|
|
|
|
|
if ( my $rest = assert $data => 'D5 00 09 04 00 11' ) { |
74
|
|
|
|
|
|
|
my $hw_ver = join('.', unpack('CCCC', $rest)); |
75
|
|
|
|
|
|
|
warn "# 3M 810 hardware version $hw_ver\n"; |
76
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
cmd( |
78
|
|
|
|
|
|
|
'13 04 01 00 02 00 03 00 04 00','FIXME: stats? rf-on?', sub { assert(shift, |
79
|
|
|
|
|
|
|
'13 00 02 01 01 03 02 02 03 00' |
80
|
|
|
|
|
|
|
)}); |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
return $hw_ver; |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
return; |
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
sub checksum { |
89
|
|
|
|
|
|
|
my $bytes = shift; |
90
|
|
|
|
|
|
|
my $crc = Digest::CRC->new( |
91
|
|
|
|
|
|
|
# midified CCITT to xor with 0xffff instead of 0x0000 |
92
|
|
|
|
|
|
|
width => 16, init => 0xffff, xorout => 0xffff, refout => 0, poly => 0x1021, refin => 0, |
93
|
|
|
|
|
|
|
) or die $!; |
94
|
|
|
|
|
|
|
$crc->add( $bytes ); |
95
|
|
|
|
|
|
|
pack('n', $crc->digest); |
96
|
|
|
|
|
|
|
} |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
sub cmd { |
99
|
|
|
|
|
|
|
my ( $hex, $description, $coderef ) = @_; |
100
|
|
|
|
|
|
|
my $bytes = hex2bytes($hex); |
101
|
|
|
|
|
|
|
if ( substr($bytes,0,1) !~ /(\xD5|\xD6)/ ) { |
102
|
|
|
|
|
|
|
my $len = pack( 'n', length( $bytes ) + 2 ); |
103
|
|
|
|
|
|
|
$bytes = $len . $bytes; |
104
|
|
|
|
|
|
|
my $checksum = checksum($bytes); |
105
|
|
|
|
|
|
|
$bytes = "\xD6" . $bytes . $checksum; |
106
|
|
|
|
|
|
|
} |
107
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
warn ">> ", as_hex( $bytes ), "\t\t[$description]\n" if $debug; |
109
|
|
|
|
|
|
|
$port->write( $bytes ); |
110
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
my $r_len = $port->read(3); |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
while ( length($r_len) < 3 ) { |
114
|
|
|
|
|
|
|
$r_len = $port->read( 3 - length($r_len) ); |
115
|
|
|
|
|
|
|
} |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
my $len = ord( substr($r_len,2,1) ); |
118
|
|
|
|
|
|
|
my $data = $port->read( $len ); |
119
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
warn "<< ", as_hex($r_len,$data), |
121
|
|
|
|
|
|
|
' | ', |
122
|
|
|
|
|
|
|
substr($data,-2,2) eq checksum(substr($r_len,1).substr($data,0,-2)) ? 'OK' : 'ERROR', |
123
|
|
|
|
|
|
|
" $len bytes\n" if $debug; |
124
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
$coderef->( $data ) if $coderef; |
127
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
} |
129
|
|
|
|
|
|
|
|
130
|
|
|
|
|
|
|
sub assert { |
131
|
|
|
|
|
|
|
my ( $got, $expected ) = @_; |
132
|
|
|
|
|
|
|
$expected = hex2bytes($expected); |
133
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
my $len = length($got); |
135
|
|
|
|
|
|
|
$len = length($expected) if length $expected < $len; |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
confess "got ", as_hex($got), " expected ", as_hex($expected) |
138
|
|
|
|
|
|
|
unless substr($got,0,$len) eq substr($expected,0,$len); |
139
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
return substr($got,$len); |
141
|
|
|
|
|
|
|
} |
142
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
sub inventory { |
145
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
my @tags; |
147
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
cmd( 'FE 00 05', 'scan for tags', sub { |
149
|
|
|
|
|
|
|
my $data = shift; |
150
|
|
|
|
|
|
|
my $rest = assert $data => 'FE 00 00 05'; |
151
|
|
|
|
|
|
|
my $nr = ord( substr( $rest, 0, 1 ) ); |
152
|
|
|
|
|
|
|
|
153
|
|
|
|
|
|
|
if ( ! $nr ) { |
154
|
|
|
|
|
|
|
warn "# no tags in range\n"; |
155
|
|
|
|
|
|
|
} else { |
156
|
|
|
|
|
|
|
my $tags = substr( $rest, 1 ); |
157
|
|
|
|
|
|
|
my $tl = length( $tags ); |
158
|
|
|
|
|
|
|
die "wrong length $tl for $nr tags: ",dump( $tags ) if $tl =! $nr * 8; |
159
|
|
|
|
|
|
|
|
160
|
|
|
|
|
|
|
foreach ( 0 .. $nr - 1 ) { |
161
|
|
|
|
|
|
|
push @tags, hex_tag substr($tags, $_ * 8, 8); |
162
|
|
|
|
|
|
|
} |
163
|
|
|
|
|
|
|
} |
164
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
}); |
166
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
warn "# tags ",dump @tags; |
168
|
|
|
|
|
|
|
return @tags; |
169
|
|
|
|
|
|
|
} |
170
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
|
172
|
|
|
|
|
|
|
# 3M defaults: 8,4 |
173
|
|
|
|
|
|
|
# cards 16, stickers: 8 |
174
|
|
|
|
|
|
|
my $max_rfid_block = 8; |
175
|
|
|
|
|
|
|
my $blocks = 8; |
176
|
|
|
|
|
|
|
|
177
|
|
|
|
|
|
|
sub _matched { |
178
|
|
|
|
|
|
|
my ( $data, $hex ) = @_; |
179
|
|
|
|
|
|
|
my $b = hex2bytes $hex; |
180
|
|
|
|
|
|
|
my $l = length($b); |
181
|
|
|
|
|
|
|
if ( substr($data,0,$l) eq $b ) { |
182
|
|
|
|
|
|
|
warn "_matched $hex [$l] in ",as_hex($data) if $debug; |
183
|
|
|
|
|
|
|
return substr($data,$l); |
184
|
|
|
|
|
|
|
} |
185
|
|
|
|
|
|
|
} |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
sub read_blocks { |
188
|
|
|
|
|
|
|
my $tag = shift || confess "no tag?"; |
189
|
|
|
|
|
|
|
$tag = shift if ref($tag); |
190
|
|
|
|
|
|
|
|
191
|
|
|
|
|
|
|
my $tag_blocks; |
192
|
|
|
|
|
|
|
my $start = 0; |
193
|
|
|
|
|
|
|
cmd( |
194
|
|
|
|
|
|
|
sprintf( "02 $tag %02x %02x", $start, $blocks ) => "read_blocks $tag $start/$blocks", sub { |
195
|
|
|
|
|
|
|
my $data = shift; |
196
|
|
|
|
|
|
|
if ( my $rest = _matched $data => '02 00' ) { |
197
|
|
|
|
|
|
|
|
198
|
|
|
|
|
|
|
my $tag = hex_tag substr($rest,0,8); |
199
|
|
|
|
|
|
|
my $blocks = ord(substr($rest,8,1)); |
200
|
|
|
|
|
|
|
warn "# response from $tag $blocks blocks ",as_hex substr($rest,9); |
201
|
|
|
|
|
|
|
foreach ( 1 .. $blocks ) { |
202
|
|
|
|
|
|
|
my $pos = ( $_ - 1 ) * 6 + 9; |
203
|
|
|
|
|
|
|
my $nr = unpack('v', substr($rest,$pos,2)); |
204
|
|
|
|
|
|
|
my $payload = substr($rest,$pos+2,4); |
205
|
|
|
|
|
|
|
warn "## pos $pos block $nr ",as_hex($payload), $/; |
206
|
|
|
|
|
|
|
$tag_blocks->{$tag}->[$nr] = $payload; |
207
|
|
|
|
|
|
|
} |
208
|
|
|
|
|
|
|
} elsif ( $rest = _matched $data => 'FE 00 00 05 01' ) { |
209
|
|
|
|
|
|
|
warn "FIXME ready? ",as_hex $rest; |
210
|
|
|
|
|
|
|
} elsif ( $rest = _matched $data => '02 06' ) { |
211
|
|
|
|
|
|
|
die "ERROR ",as_hex($rest); |
212
|
|
|
|
|
|
|
} else { |
213
|
|
|
|
|
|
|
die "FIXME unsuported ",as_hex($rest); |
214
|
|
|
|
|
|
|
} |
215
|
|
|
|
|
|
|
}); |
216
|
|
|
|
|
|
|
|
217
|
|
|
|
|
|
|
warn "# tag_blocks ",dump($tag_blocks); |
218
|
|
|
|
|
|
|
return $tag_blocks; |
219
|
|
|
|
|
|
|
} |
220
|
|
|
|
|
|
|
|
221
|
|
|
|
|
|
|
sub write_blocks { |
222
|
|
|
|
|
|
|
my $tag = shift; |
223
|
|
|
|
|
|
|
$tag = shift if ref $tag; |
224
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
my $data = shift; |
226
|
|
|
|
|
|
|
$data = join('', @$data) if ref $data eq 'ARRAY'; |
227
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
warn "## write_blocks ",dump($tag,$data); |
229
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
if ( length($data) % 4 ) { |
231
|
|
|
|
|
|
|
$data .= '\x00' x ( 4 - length($data) % 4 ); |
232
|
|
|
|
|
|
|
warn "# padded data to ",dump($data); |
233
|
|
|
|
|
|
|
} |
234
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
my $hex_data = as_hex $data; |
236
|
|
|
|
|
|
|
my $blocks = sprintf('%02x', length($data) / 4 ); |
237
|
|
|
|
|
|
|
|
238
|
|
|
|
|
|
|
cmd( |
239
|
|
|
|
|
|
|
"04 $tag 00 $blocks 00 $hex_data", "write_blocks $tag [$blocks] $hex_data", sub { |
240
|
|
|
|
|
|
|
my $data = shift; |
241
|
|
|
|
|
|
|
if ( my $rest = _matched $data => '04 00' ) { |
242
|
|
|
|
|
|
|
my $tag = substr($rest,0,8); |
243
|
|
|
|
|
|
|
my $blocks = substr($rest,8,1); |
244
|
|
|
|
|
|
|
warn "# WRITE ",as_hex($tag), " [$blocks]\n"; |
245
|
|
|
|
|
|
|
} elsif ( $rest = _matched $data => '04 06' ) { |
246
|
|
|
|
|
|
|
die "ERROR ",as_hex($rest); |
247
|
|
|
|
|
|
|
} else { |
248
|
|
|
|
|
|
|
die "UNSUPPORTED"; |
249
|
|
|
|
|
|
|
} |
250
|
|
|
|
|
|
|
} |
251
|
|
|
|
|
|
|
); |
252
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
} |
254
|
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
sub read_afi { |
256
|
|
|
|
|
|
|
my $tag = shift; |
257
|
|
|
|
|
|
|
$tag = shift if ref $tag; |
258
|
|
|
|
|
|
|
|
259
|
|
|
|
|
|
|
my $afi; |
260
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
cmd( |
262
|
|
|
|
|
|
|
"0A $tag", "read_afi $tag", sub { |
263
|
|
|
|
|
|
|
my $data = shift; |
264
|
|
|
|
|
|
|
|
265
|
|
|
|
|
|
|
if ( my $rest = _matched $data => '0A 00' ) { |
266
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
my $tag = substr($rest,0,8); |
268
|
|
|
|
|
|
|
$afi = substr($rest,8,1); |
269
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
warn "# SECURITY ", hex_tag($tag), " AFI: ", as_hex($afi); |
271
|
|
|
|
|
|
|
|
272
|
|
|
|
|
|
|
} elsif ( $rest = _matched $data => '0A 06' ) { |
273
|
|
|
|
|
|
|
die "ERROR reading security from $tag ", as_hex($data); |
274
|
|
|
|
|
|
|
} else { |
275
|
|
|
|
|
|
|
die "IGNORED ",as_hex($data); |
276
|
|
|
|
|
|
|
} |
277
|
|
|
|
|
|
|
}); |
278
|
|
|
|
|
|
|
warn "## read_afi ",dump($tag, $afi); |
279
|
|
|
|
|
|
|
return $afi; |
280
|
|
|
|
|
|
|
} |
281
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
sub write_afi { |
283
|
|
|
|
|
|
|
my $tag = shift; |
284
|
|
|
|
|
|
|
$tag = shift if ref $tag; |
285
|
|
|
|
|
|
|
my $afi = shift || die "no afi?"; |
286
|
|
|
|
|
|
|
|
287
|
|
|
|
|
|
|
$afi = as_hex $afi; |
288
|
|
|
|
|
|
|
|
289
|
|
|
|
|
|
|
cmd( |
290
|
|
|
|
|
|
|
"09 $tag $afi", "write_afi $tag $afi", sub { |
291
|
|
|
|
|
|
|
my $data = shift; |
292
|
|
|
|
|
|
|
|
293
|
|
|
|
|
|
|
if ( my $rest = _matched $data => '09 00' ) { |
294
|
|
|
|
|
|
|
my $tag_back = hex_tag substr($rest,0,8); |
295
|
|
|
|
|
|
|
die "write_afi got $tag_back expected $tag" if $tag_back ne $tag; |
296
|
|
|
|
|
|
|
warn "# SECURITY ", hex_tag($tag), " AFI: ", as_hex($afi); |
297
|
|
|
|
|
|
|
} elsif ( $rest = _matched $data => '0A 06' ) { |
298
|
|
|
|
|
|
|
die "ERROR writing AFI to $tag ", as_hex($data); |
299
|
|
|
|
|
|
|
} else { |
300
|
|
|
|
|
|
|
die "IGNORED ",as_hex($data); |
301
|
|
|
|
|
|
|
} |
302
|
|
|
|
|
|
|
}); |
303
|
|
|
|
|
|
|
warn "## write_afi ", dump( $tag, $afi ); |
304
|
|
|
|
|
|
|
return $afi; |
305
|
|
|
|
|
|
|
} |
306
|
|
|
|
|
|
|
|
307
|
|
|
|
|
|
|
1 |
308
|
|
|
|
|
|
|
|
309
|
|
|
|
|
|
|
__END__ |