2000-11-09 16:20:37 ShengHuo ZHU <zsh@cs.rochester.edu>
[gnus] / lisp / binhex.el
1 ;;; binhex.el -- elisp native binhex decode
2 ;; Copyright (c) 1998 Free Software Foundation, Inc.
3
4 ;; Author: Shenghuo Zhu <zsh@cs.rochester.edu>
5 ;; Create Date: Oct 1, 1998
6 ;; Keywords: binhex news
7
8 ;; This file is part of GNU Emacs.
9
10 ;; GNU Emacs is free software; you can redistribute it and/or modify
11 ;; it under the terms of the GNU General Public License as published by
12 ;; the Free Software Foundation; either version 2, or (at your option)
13 ;; any later version.
14
15 ;; GNU Emacs is distributed in the hope that it will be useful,
16 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18 ;; GNU General Public License for more details.
19
20 ;; You should have received a copy of the GNU General Public License
21 ;; along with GNU Emacs; see the file COPYING.  If not, write to the
22 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
23 ;; Boston, MA 02111-1307, USA.
24
25 ;;; Commentary:
26
27 ;;; Code:
28
29 (eval-when-compile (require 'cl))
30
31 (if (not (fboundp 'char-int))
32     (defalias 'char-int 'identity))
33
34 (defvar binhex-decoder-program "hexbin"
35   "*Non-nil value should be a string that names a uu decoder.
36 The program should expect to read binhex data on its standard
37 input and write the converted data to its standard output.")
38
39 (defvar binhex-decoder-switches '("-d")
40   "*List of command line flags passed to the command named by binhex-decoder-program.")
41
42 (defconst binhex-alphabet-decoding-alist
43   '(( ?\! . 0) ( ?\" . 1) ( ?\# . 2) ( ?\$ . 3) ( ?\% . 4) ( ?\& . 5)
44     ( ?\' . 6) ( ?\( . 7) ( ?\) . 8) ( ?\* . 9) ( ?\+ . 10) ( ?\, . 11)
45     ( ?\- . 12) ( ?0 . 13) ( ?1 . 14) ( ?2 . 15) ( ?3 . 16) ( ?4 . 17)
46     ( ?5 . 18) ( ?6 . 19) ( ?8 . 20) ( ?9 . 21) ( ?@ . 22) ( ?A . 23)
47     ( ?B . 24) ( ?C . 25) ( ?D . 26) ( ?E . 27) ( ?F . 28) ( ?G . 29)
48     ( ?H . 30) ( ?I . 31) ( ?J . 32) ( ?K . 33) ( ?L . 34) ( ?M . 35)
49     ( ?N . 36) ( ?P . 37) ( ?Q . 38) ( ?R . 39) ( ?S . 40) ( ?T . 41)
50     ( ?U . 42) ( ?V . 43) ( ?X . 44) ( ?Y . 45) ( ?Z . 46) ( ?\[ . 47)
51     ( ?\` . 48) ( ?a . 49) ( ?b . 50) ( ?c . 51) ( ?d . 52) ( ?e . 53)
52     ( ?f . 54) ( ?h . 55) ( ?i . 56) ( ?j . 57) ( ?k . 58) ( ?l . 59)
53     ( ?m . 60) ( ?p . 61) ( ?q . 62) ( ?r . 63)))
54
55 (defun binhex-char-map (char)
56   (cdr (assq char binhex-alphabet-decoding-alist)))
57
58 ;;;###autoload
59 (defconst binhex-begin-line
60   "^:...............................................................$")
61 (defconst binhex-body-line
62   "^[^:]...............................................................$")
63 (defconst binhex-end-line ":$")
64
65 (defvar binhex-temporary-file-directory
66   (cond ((fboundp 'temp-directory) (temp-directory))
67         ((boundp 'temporary-file-directory) temporary-file-directory)
68         ("/tmp/")))
69
70 (if (featurep 'xemacs)
71     (defalias 'binhex-insert-char 'insert-char)
72   (defun binhex-insert-char (char &optional count ignored buffer)
73     (if (or (null buffer) (eq buffer (current-buffer)))
74         (insert-char char count)
75       (with-current-buffer buffer
76         (insert-char char count)))))
77
78 (defvar binhex-crc-table
79   [0  4129  8258  12387  16516  20645  24774  28903
80       33032  37161  41290  45419  49548  53677  57806  61935
81       4657  528  12915  8786  21173  17044  29431  25302
82       37689  33560  45947  41818  54205  50076  62463  58334
83       9314  13379  1056  5121  25830  29895  17572  21637
84       42346  46411  34088  38153  58862  62927  50604  54669
85       13907  9842  5649  1584  30423  26358  22165  18100
86       46939  42874  38681  34616  63455  59390  55197  51132
87       18628  22757  26758  30887  2112  6241  10242  14371
88       51660  55789  59790  63919  35144  39273  43274  47403
89       23285  19156  31415  27286  6769  2640  14899  10770
90       56317  52188  64447  60318  39801  35672  47931  43802
91       27814  31879  19684  23749  11298  15363  3168  7233
92       60846  64911  52716  56781  44330  48395  36200  40265
93       32407  28342  24277  20212  15891  11826  7761  3696
94       65439  61374  57309  53244  48923  44858  40793  36728
95       37256  33193  45514  41451  53516  49453  61774  57711
96       4224  161  12482  8419  20484  16421  28742  24679
97       33721  37784  41979  46042  49981  54044  58239  62302
98       689  4752  8947  13010  16949  21012  25207  29270
99       46570  42443  38312  34185  62830  58703  54572  50445
100       13538  9411  5280  1153  29798  25671  21540  17413
101       42971  47098  34713  38840  59231  63358  50973  55100
102       9939  14066  1681  5808  26199  30326  17941  22068
103       55628  51565  63758  59695  39368  35305  47498  43435
104       22596  18533  30726  26663  6336  2273  14466  10403
105       52093  56156  60223  64286  35833  39896  43963  48026
106       19061  23124  27191  31254  2801  6864  10931  14994
107       64814  60687  56684  52557  48554  44427  40424  36297
108       31782  27655  23652  19525  15522  11395  7392  3265
109       61215  65342  53085  57212  44955  49082  36825  40952
110       28183  32310  20053  24180  11923  16050  3793  7920])
111
112 (defun binhex-update-crc (crc char &optional count)
113   (if (null count) (setq count 1))
114   (while (> count 0)
115     (setq crc (logxor (logand (lsh crc 8) 65280)
116                       (aref binhex-crc-table
117                             (logxor (logand (lsh crc -8) 255)
118                                     char)))
119           count (1- count)))
120   crc)
121
122 (defun binhex-verify-crc (buffer start end)
123   (with-current-buffer buffer
124     (let ((pos start) (crc 0) (last (- end 2)))
125       (while (< pos last)
126         (setq crc (binhex-update-crc crc (char-after pos))
127               pos (1+ pos)))
128       (if (= crc (binhex-string-big-endian (buffer-substring last end)))
129           nil
130         (error "CRC error")))))
131
132 (defun binhex-string-big-endian (string)
133   (let ((ret 0) (i 0) (len (length string)))
134     (while (< i len)
135       (setq ret (+ (lsh ret 8) (char-int (aref string i)))
136             i (1+ i)))