forked from agalakhov/captdriver
-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathSPECS
414 lines (338 loc) · 14.3 KB
/
SPECS
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
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
This document describes printing protocol of Canon CAPT printers.
Written by Alexey Galakhov <agalakhov@gmail.com>
Reverse engineering done by:
2004 Nicolas Boichat (LBP-810, based on Rildo Pragana's Samsung ML-85P work)
2010 Alexey Galakhov (LBP-2900)
2010 Benoit Bolsee (LBP-3010)
2013 Alexey Galakhov - summarized everything and corrected errors
1. A0-command protocol
======================
The same high level protocol called "CAPT" seems to be used by all Canon LBP
printers, both LPT and USB. The exact command set depends on the printer model.
The protocol is essentially bi-directional. The same command set is used in the
unidirectional mode for communication between Canon proprietary utility
"captfilter" and the printer port monitor. This commuinication uses an
extended command set that is translated to the printer command set somewhere
inside "ccpd" daemon.
1.1. General format of CAPT commands
------------------------------------
All transmit are in 16-bit, little-endian words. Each command has the HEADER
of 4 bytes followed:
struct capt_packet {
uint16_t command; /* command code */
uint16_t size; /* size of the transmit, incl. header */
uint16_t payload[(size - 4) / 2]; /* data, 0 or more bytes */
};
Example: command 0xD0A1 with no payload is encoded as follows: A1 D0 04 00.
Some commands have replies. A printer reply has the same format as the command.
The command code in the reply is the same as in the command.
Example:
command: A5 E0 14 00 EE DB EA AD 00 00 00 00 00 00 00 00 00 00 00 00
reply: A5 E0 06 00 00 00
If the command requires a reply, the driver MUST read the reply before issuing
any other commands. Failure to do so will cause printer deadlock. If the command
requires no reply, the driver should not try to wait to the reply or else it
will wait forever.
If the reply has size of more than 6 bytes, it is split into two packets, the
one of 6 bytes and the one with the rest. No replies are shorter than 6 bytes.
Quirk: some printer models reply to some commands with size encoded as BCD
instead of normal binary code. This is obviously a bug in the printer firmware.
Example: decimal 57 may be encoded as 0x57 0x00 and not as 0x39 0x00.
1.2. Common printer control commands
------------------------------------
* 0xA0A0
NOP? Get status?
* 0xA0A1
Get printer status. See section 1.3.
* 0xA0A8
Get printer extended status. See section 1.3.
* 0xA1A0
Get printer identification string (IEEE-1284). This command does NOT use the
common reply format and just returns raw string data.
* 0xA1A1
Get printer capabitilies?
1.3. Printer status
-------------------
Extended status record (reply to the 0xA0A8 command):
0-1: uint16_t bits STATUS0
2-7: ? 00 00 0f 00 00 00 (constant sequence)
8-9: uint16_t bits STATUS1
10-11: uint16_t bits STATUS2
12-13: uint16_t bits STATUS3
14-15: uint16_t page number being decoded
16-17: uint16_t page number being printed
18-19: uint16_t page number being pushed out
20-21: uint16_t page number completed
22-23: 00 00
24-25: uint16_t bits STATUS4
26-27: 00 00
28-29: uint16_t job number being processed
30-31: uint16_t bits STATUS5
32: ? 55
33: uint8_t number of 0xE0A5 commands sent
34-35: uint16_t number of received pages
36-37: 00 00
38-39: uint16_t bits STATUS6
40-41: uint16_t number of pages in job (set by user 0xE1A1)
42-43: 00 00
44-45: uint16_t number of page printed in job
46-63: 00 00 00 00 00 00 00 00 01 00 00 00 00 00 00 00 00 00
64-65: uint16_t event counter
66-83: 00 00 00 ...
Status bits:
STATUS0
15: ? 0 initially, 1 after UPLOAD_0
14-12: ? 0 0 0
11: ? 0 initially, 1 after UPLOAD_0
10: ?
9: 0xA0A1 result changed
8: XSTATUS changed
7: Busy, try to get status again
6: ? 0
5-4: Uninitialized (not ready), reset by 0xE0A5
3: ?
2: 1 = buffer full (do not send data), 0 = buffer empty
1: Paper out (reset by Cancel or auto)
0: Processing job
STATUS1
15: 0
14: Paper out (reset by Reset)
13-8: 0
7: Processing data
6-3: ? 0
2: Printing
1: ? 0
0: Powering up
STATUS2
15-9: 0
8: Paper button pressed
7: Problem
6-0: 0
STATUS3
15-13: 0
12: Powering up
11-0: 0
STATUS4
15-8: 0
7: 0
6: 1
5: Button blinking
4: 1
3:
2:
1: 0 ready, 1 stop
0: Button active
STATUS5
STATUS6
(to be done)
1.4. Compression control commands
---------------------------------
All 0xD0xx commands have no reply. They all are one-way.
* 0xD0A0
Page parameters. Mandatory. Issued before each page.
0-1: ? 00 00
2-3: ? 30 2a
^^ ^^ fixed vaules, differs with model:
30 2a lpb2900
?? ?? lpb3000
ba 09 lbp3010
4-7: ? 02 00 00 00
^^ related to page dimensions:
media size lpb2900 lpb3000 lpb3010
A4 (default) ?? ?? 02
A5 ?? ?? 03
B5 ?? ?? 07
Executive ?? ?? 0a
Legal ?? ?? 0c
Letter ?? ?? 0d
EnvC5 ?? ?? 15
Env10 ?? ?? 16
EnvMonarch ?? ?? 17
EnvDL ?? ?? 18
3x5 ?? ?? 40
PRC16K ?? ?? d4
8-11: 1c 1c 1c 1c
^^ toner density (black): only bits from 2 to 5 are used, other never change:
7-6: 0
5-2: exadecimal value of the blak toner density
1-0: 0 (1 for lbp3010, but also works with 0)
Toner density range changes from model to model:
0-7 lbp2900 (3 bits used)
0-7 lbp3000 (3 bits used)
0-15 lbp3010 (4 bits used)
Dafaul values:
1c lbp2900 (bits 0,1,7,8 are always 0)
?? lbp3000 (??)
1f lbp3010 (bits 0,1 are always 1, bits 7,8 are always 0)
Bytes from 9 to 11 could be used in color printers (maybe for yellow,magenta and cyan)
12: paper type, often this requires other bytes changes:
media type lpb2900 lpb3000 lpb3010
Plain 00 ?? 01
Plain L ?? ?? 08
Thick ?? ?? 04*
Thick H ?? ?? 05*
Transparency 24 ?? 24
Label ?? ?? 04*
Envelope 20 ?? 20*
media size lpb2900 lpb3000 lpb3010
A4 (default) ?? ?? ??
A5 ?? ?? 21*
B5 ?? ?? 21*
Executive ?? ?? 21*
Legal ?? ?? ??
Letter ?? ?? ??
EnvC5 ?? ?? 21*
Env10 ?? ?? 21*
EnvMonarch ?? ?? 21*
EnvDL ?? ?? 21*
3x5 ?? ?? 21*
PRC16K ?? ?? ??
*lbp3010: also set byte 13 at 11
30 ? Special Print Mode (see byte 34)
13: ? related to page size, maybe adapt to size:
11 ? default for lbp2900, in lbpp3010 used with all page size except Legal, 16k, A4, Letter
81 ? default for lbp3010, never seen in lbp2900
14-18: ? 04 00 01 01 02
19: toner saving (0 = no, 1 = yes). Unused with some model such as in lbp3010 for which toner density setting is effective.
20-21: ? 00 00
22-23: uint16_t image margin height, lower first.
24-25: uint16_t image margin width, lower first.
26-27: uint16_t image width (line size in bytes) = LINESIZE for compressor
28-29: uint16_t image height (number of lines)
30-31: uint16_t paper width (pixels = inches * 600)
32-33: uint16_t paper height (pixels = inches * 600)
-- Older printers have only 34 bytes
34-35: ? 00 00
^^ in lbp3010 is 01 with Special Print Mode
36: ? fuser mode:
media type lpb2900 lpb3000 lpb3010
Plain (default) 01 ?? 01
Plain L 01 ?? 01
Thick 01 ?? 02
Thick H 02 ?? 02
Transparency 13 ?? 13
Label ?? ?? 14
Envelope 1c ?? 1c
37-39: ? 00 00 00
* 0xD0A1
Initialize page? Has no data. Issued before each page.
* 0xD0A2
Reset? Has no data. Issued after page but may be issued before page.
* 0xD0A4
Hi-SCoA compression parameters on Hi-SCoA printers. Mandatory. Issued before
sending Hi-SCoA compressed data. See section 3 for the complete description of
these parameters. Contains 8-byte data.
0: int8_t value of L3, positive
1: int8_t value of L5, positive
2: 0x01 unknown, may be flag
3: 0x01 unknown, may be bpp (0x08 for color printers?)
4: int8_t value of L0 (?), always zero
5: int8_t value of L2, negative
6-7: int16_t value of L4 (?)
* 0xD0A5, 0xD0A6, 0xD0A7
Hi-SCoA parameters for color printers?
* 0xD0A9 - multi-command
Contains other 0xD0nn commands, like that: A9 D0 C0 00 A1 D0 04 00 A2 D0 04 00
Used to send multiple 0xD0nn commands in a row on printers that support it.
2. SCoA compression algorithm
=============================
To be written.
3. Hi-SCoA compression algorithm
================================
Hi-SCoA data are compressed band by band, each band is compressed independently
of others. Band size may vary. It seems that most printers may work with
different band sizes. The algorithm is known completely.
3.1. Data encoding
------------------
Hi-SCoA data are represented as a bit stream. Bits are written MSB first. The
entire stream is always 4-byte aligned by adding 1s up to 32-bit boundary. On a
little-endian machine the stream is written by bytes and not by 32-bit words.
This does not matter on big-endian.
Example: bits 1,0,0,0,1,1,0,1,1,1,1,0,1,0,1,1 correspond to 8D EB FF FF.
The encoded stream is obfuscated by XORing it with the constant value 0x43.
The above example becomes CE A8 BC BC.
3.2. Command encoding
---------------------
Hi-SCoA bit stream consists of commands. Command opcodes are written using
unary coding. There are 8 commands. Some commands have two subcommands which are
encoded by adding one bit after the command code.
Command 0: LONGREP0
Command 10: REPBYTE
Command 110:
subcommand 1: BYTE
subcommand 0: LONGREP2
Command 1110: LONGREP3
Command 11110: LONGREP4
Command 111110: LONGREP5
Command 1111110:
subcommand 1: ZEROBYTE
subcommand 0: PREFIX
Command 11111110: END
Bit sequence 11111111 used for alignment only and seems to be NOP.
3.3. Number encoding
--------------------
LONGREP commands use number parameter that is encoded in Elias gamma coding.
First, order is encoded by putting [order] 1s and one 0 to the stream. The order
is always 5 or less. Then [order+1] bits of the number N follow (MSB first).
Result is given by the formula: number = (1 << (order + 2)) - 1 - N.
There are exceptions from this rule. Zero is encoded as 111111. If order is 0,
it is followed by one or two bits: 0 -> 1, 11 -> 2, 10 -> 3.
Example:
0 111111
1 00
2 011
3 010
4 1011
5 1010
6 1001
7 1000
8 110111
15 110000
16 11101111
17 11101110
etc.
Numbers of 128 or more are not encoded this way.
PREFIX command uses number parameter that is encoded in the following coding.
First, order is written in two bits (MSB first). Then [order] bits of N follow.
The resulting value is given by: prefix = 128 * ((1 << order) - 1 - N).
Numbers of more than 512 are not encoded this way.
3.4. Command description
------------------------
Hi-SCoA uses a significantly simplified and limited variant of LZ77 compression.
In addition to the usual window it maintains a stash of no more than 16 bytes.
A byte may be pushed to the stash (the stash is shifted so that the last byte
has now index 0, like a stack). A byte may be popped from the stash by its
index. It is then always pushed back and becomes index 0.
The stream consists of the following commands:
BYTE - read the following 8 bits, put the corresponding byte to the stream
and push it to the stack
ZEROBYTE - shorter equivalent of BYTE 00000000
REPBYTE - read the following 4 bits (i), take and remove [15-i]th byte from
the stash, put it to the stream and push back to the stash.
PREFIX - read the following prefix (see section 3.3) and apply it to the
next command. Used only before LONGREPx.
LONGREPx - read the following number N (see section 3.3) and copy prefix+N
bytes from output position POSx to the output. The following
positions are initially used:
POS0 = LINESIZE + L0
POS2 = LINESIZE + L2
POS3 = L3
POS4 = L4
POS5 = L5
In addition, LONGREP2 exchanges L2 and L0 and LONGREP5 exchanges
L3 and L5 after copying. (This may be configurable, however).
LONGREP commands in Canon driver never cross line boundary but this
seems to be optional and not really needed.
END - read the following 2-bit number and stop decompression. Number
meanings are:
00 - end of band
01 - end of page
10 - unknown/never seen
11 - unknown/never seen
Compression constants L0, L2, L3, L4, L5 and LINESIZE are sent to the
decompressor in 0xD0A0 and 0xD0A4 commands (see section 1.4). L0 is always 0,
other values are typically
L2 = -7, L3 = 1, L4 = (not used), L5 = 4, LINESIZE = 592 or 608
L2 = -5, L3 = 4, L4 = 384, L5 = 12, LINESIZE = 592
The actual values are probably optimized for certain dithering algorithm. They
rarely change but are not constant for the printer.