summary refs log tree commit diff
path: root/mrf.html
blob: 7415980e2b5822f477ed56f4d19a6c48731889e1 (plain) (blame)
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
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.3//EN">
<html><head><title>MRF image format specification</title></head>
<body>
<a href="#index">Table Of Contents</a>

<h1>MRF format</h1>
Updated: 1991
<br>
<h2>NAME</h2>

MRF - monochrome recursive format (compressed bitmaps)

<h2 id="description">DESCRIPTION</h2>

<p>This document describes the MRF format recognized by
<a href="index.html">Netpbm</a>.

<p>MRF is a compressed format for bilevel (1-bit mono) images.  It
achieves better compression for some such images than either GIF or
PNG. (It's also very easy to implement (about the same difficulty as
RLE, I'd say) and an MRF reader needs no tables/buffers, which may
make it useful for tiny machines).

<p>In case the above hasn't made it sufficiently clear, I'll make this
next point explicitly: <em>MRF cannot represent color at all.</em> Nor
can it represent grayscale.  It's a specifically mono format.  (If you
want to compress a color or grayscale image, my advice is to use
JPEG2000).

<p>First, here's what goes where in an MRF file. I'll explain how the
compression works afterward.

<dl compact>
<dt>Offset<dd>
Description
<dt>0
<dd>
magic number - "MRF1" (in ASCII)

<dt>4
<dd>
width (32-bit, MSB first (i.e. big-endian))

<dt>8
<dd>
height (same)

<dt>12
<dd>
reserved (single byte, must be zero)

<dt>13
<dd>
compressed data

</dl>

<p>Note that there is no end-of-file marker in the file itself - the
compressed data carries on right up to EOF.

<p>The way the picture is compressed is essentially very simple, but
as they say, the devil is in the detail.  So don't be put off if it
sounds confusing.

<p>The image is treated as a number of 64x64 squares, forming a grid
large enough to encompass it. (If an image is (say) 129x65, it'll be
treated in the same way as a 192x128 one. On decompression, the extra
area which was encoded (the contents of this area is undefined) should
be ignored.) Each of these squares in turn (in left-to-right,
top-to-bottom order) is recursively subdivided until the smallest
completely black or white squares are found. Some pseudocode (eek!)
for the recursive subdivision routine should make things clearer:

<pre>
    if square size &gt; 1x1 and square is all one color, output 1 bit
    if whole square is black, output a 0 bit and return
    if whole square is white, output a 1 bit and return
    output a 0 bit
    divide the square into four quarters, calling routine for
    each in this order: top-left, top-right, bottom-left, bottom-right
</pre>

<p>(Note that the "output a 0 bit" stage is not reached for squares
of size 1x1, which is what stops it recursing infinitely.  I mention
this as it may not be immediately obvious.)

<p>The whole of the compressed data is made up of the bits output by
the above routine. The bits are packed into bytes MSB first, so for
example outputting the bits 1,0,0,0,0,0,0,0 would result in a 80h byte
being output. Any `unused' bits in the last byte output are undefined;
these are effectively after EOF and their value is unimportant.

<p>If writing that sounds too much like hard work :-), you could
always adapt <b>pbmtomrf</b> and/or <b>mrftopbm</b>.  That's the main
reason their source code is public domain, after all.

<p>Above, I said the contents of any extra area encoded (when a bitmap
smaller than the grid of squares is compressed) is undefined.  This is
deliberate so that the MRF compressor can make these unseen areas
anything it wants so as to maximize compression, rather than simply
leaving it blank. <b>pbmtomrf</b> does a little in this respect but
could definitely be improved upon.

<p><b>mrftopbm</b>'s <b>-1</b> option causes it to include the edges, if
any, in the output PBM.  This may help when debugging a compressor's
edge optimization.

<p>Note that the "F" in the name "MRF" comes from "format," which is redundant
because it is the name of a format.  That sort of makes "MRF format" sound
as stupid as "PIN number," but it's not really that bad.

<h2 id="seealso">SEE ALSO</h2>

<b><a href="mrftopbm.html">mrftopbm</a></b>,
<b><a href="pbmtomrf.html">pbmtomrf</a></b>

<h2 id="author">AUTHOR</h2>

Russell Marks.

<hr>
<h2 id="index">Table Of Contents</h2>
<ul>
<li><a href="#description">DESCRIPTION</a>
<li><a href="#seealso">SEE ALSO</a>
<li><a href="#author">AUTHOR</a>
</ul>
</body>
</html>