rot13fs.c revision 1.4 1 1.4 pooka /* $NetBSD: rot13fs.c,v 1.4 2007/04/12 15:09:02 pooka Exp $ */
2 1.1 pooka
3 1.1 pooka /*
4 1.1 pooka * Copyright (c) 2007 Antti Kantee. All Rights Reserved.
5 1.1 pooka *
6 1.1 pooka * Redistribution and use in source and binary forms, with or without
7 1.1 pooka * modification, are permitted provided that the following conditions
8 1.1 pooka * are met:
9 1.1 pooka * 1. Redistributions of source code must retain the above copyright
10 1.1 pooka * notice, this list of conditions and the following disclaimer.
11 1.1 pooka * 2. Redistributions in binary form must reproduce the above copyright
12 1.1 pooka * notice, this list of conditions and the following disclaimer in the
13 1.1 pooka * documentation and/or other materials provided with the distribution.
14 1.1 pooka * 3. The name of the company nor the name of the author may be used to
15 1.1 pooka * endorse or promote products derived from this software without specific
16 1.1 pooka * prior written permission.
17 1.1 pooka *
18 1.1 pooka * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
19 1.1 pooka * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 1.1 pooka * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 1.1 pooka * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22 1.1 pooka * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 1.1 pooka * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
24 1.1 pooka * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 1.1 pooka * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26 1.1 pooka * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27 1.1 pooka * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 1.1 pooka * SUCH DAMAGE.
29 1.1 pooka */
30 1.1 pooka
31 1.1 pooka /*
32 1.1 pooka * rot13fs: puffs layering experiment
33 1.1 pooka * (unfinished, as is probably fairly easy to tell)
34 1.1 pooka */
35 1.1 pooka
36 1.1 pooka #include <err.h>
37 1.1 pooka #include <errno.h>
38 1.1 pooka #include <puffs.h>
39 1.1 pooka #include <stdio.h>
40 1.1 pooka #include <stdlib.h>
41 1.1 pooka #include <unistd.h>
42 1.1 pooka
43 1.1 pooka PUFFSOP_PROTOS(puffs_null) /* XXX */
44 1.1 pooka PUFFSOP_PROTOS(rot13)
45 1.1 pooka
46 1.1 pooka static void usage(void);
47 1.1 pooka
48 1.1 pooka static void
49 1.1 pooka usage()
50 1.1 pooka {
51 1.1 pooka
52 1.1 pooka errx(1, "usage: %s [-s][-o mntopts]rot13path mountpath",
53 1.1 pooka getprogname());
54 1.1 pooka }
55 1.1 pooka
56 1.1 pooka static uint8_t tbl[256];
57 1.1 pooka
58 1.1 pooka static void
59 1.1 pooka dorot13(void *buf, size_t buflen)
60 1.1 pooka {
61 1.1 pooka uint8_t *b = buf;
62 1.1 pooka
63 1.1 pooka while (buflen--) {
64 1.1 pooka *b = tbl[*b];
65 1.1 pooka b++;
66 1.1 pooka }
67 1.1 pooka }
68 1.1 pooka
69 1.1 pooka static int
70 1.1 pooka rot13path(struct puffs_usermount *pu, struct puffs_pathobj *base,
71 1.1 pooka struct puffs_cn *pcn)
72 1.1 pooka {
73 1.1 pooka
74 1.1 pooka dorot13(pcn->pcn_name, pcn->pcn_namelen);
75 1.1 pooka
76 1.1 pooka return 0;
77 1.1 pooka }
78 1.1 pooka
79 1.1 pooka int
80 1.1 pooka main(int argc, char *argv[])
81 1.1 pooka {
82 1.1 pooka struct puffs_usermount *pu;
83 1.1 pooka struct puffs_ops *pops;
84 1.1 pooka struct puffs_pathobj *po_root;
85 1.4 pooka struct puffs_node *pn_root;
86 1.1 pooka struct statvfs svfsb;
87 1.1 pooka struct stat sb;
88 1.1 pooka mntoptparse_t mp;
89 1.1 pooka int mntflags, pflags, lflags;
90 1.1 pooka int ch;
91 1.1 pooka int i;
92 1.1 pooka
93 1.1 pooka setprogname(argv[0]);
94 1.1 pooka
95 1.1 pooka if (argc < 3)
96 1.1 pooka usage();
97 1.1 pooka
98 1.1 pooka pflags = lflags = mntflags = 0;
99 1.1 pooka while ((ch = getopt(argc, argv, "o:s")) != -1) {
100 1.1 pooka switch (ch) {
101 1.1 pooka case 'o':
102 1.1 pooka mp = getmntopts(optarg, puffsmopts, &mntflags, &pflags);
103 1.1 pooka if (mp == NULL)
104 1.1 pooka err(1, "getmntopts");
105 1.1 pooka freemntopts(mp);
106 1.1 pooka break;
107 1.1 pooka case 's':
108 1.1 pooka lflags |= PUFFSLOOP_NODAEMON;
109 1.1 pooka break;
110 1.1 pooka }
111 1.1 pooka }
112 1.1 pooka pflags |= PUFFS_FLAG_BUILDPATH;
113 1.1 pooka argv += optind;
114 1.1 pooka argc -= optind;
115 1.1 pooka
116 1.1 pooka if (pflags & PUFFS_FLAG_OPDUMP)
117 1.1 pooka lflags = PUFFSLOOP_NODAEMON;
118 1.1 pooka
119 1.1 pooka if (argc != 2)
120 1.1 pooka usage();
121 1.1 pooka
122 1.1 pooka PUFFSOP_INIT(pops);
123 1.1 pooka
124 1.1 pooka PUFFSOP_SET(pops, puffs_null, fs, statvfs);
125 1.1 pooka PUFFSOP_SETFSNOP(pops, unmount);
126 1.1 pooka PUFFSOP_SETFSNOP(pops, sync);
127 1.1 pooka
128 1.1 pooka PUFFSOP_SET(pops, rot13, node, readdir);
129 1.1 pooka PUFFSOP_SET(pops, rot13, node, read);
130 1.1 pooka PUFFSOP_SET(pops, rot13, node, write);
131 1.1 pooka
132 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, lookup);
133 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, create);
134 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, mknod);
135 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, getattr);
136 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, setattr);
137 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, fsync);
138 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, remove);
139 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, link);
140 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, rename);
141 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, mkdir);
142 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, rmdir);
143 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, symlink);
144 1.1 pooka PUFFSOP_SET(pops, puffs_null, node, readlink);
145 1.2 pooka PUFFSOP_SET(pops, puffs_null, node, reclaim);
146 1.1 pooka
147 1.1 pooka if ((pu = puffs_mount(pops, argv[1], mntflags, "rot13", NULL,
148 1.1 pooka pflags, 0)) == NULL)
149 1.1 pooka err(1, "mount");
150 1.1 pooka
151 1.1 pooka if (statvfs(argv[0], &svfsb) == -1)
152 1.1 pooka err(1, "statvfs %s", argv[0]);
153 1.1 pooka
154 1.4 pooka pn_root = puffs_pn_new(pu, NULL);
155 1.4 pooka if (pn_root == NULL)
156 1.1 pooka err(1, "puffs_pn_new");
157 1.4 pooka puffs_setroot(pu, pn_root);
158 1.1 pooka
159 1.1 pooka po_root = puffs_getrootpathobj(pu);
160 1.1 pooka if (po_root == NULL)
161 1.1 pooka err(1, "getrootpathobj");
162 1.1 pooka po_root->po_path = argv[0];
163 1.1 pooka po_root->po_len = strlen(argv[0]);
164 1.1 pooka
165 1.1 pooka puffs_set_namemod(pu, rot13path);
166 1.1 pooka
167 1.1 pooka if (stat(argv[0], &sb) == -1)
168 1.1 pooka err(1, "stat %s", argv[0]);
169 1.4 pooka puffs_stat2vattr(&pn_root->pn_va, &sb);
170 1.1 pooka
171 1.1 pooka /* initialize rot13 tables */
172 1.1 pooka for (i = 0; i < 256; i++)
173 1.1 pooka tbl[i] = (uint8_t)i;
174 1.1 pooka for (i = 0; i <= 26; i++)
175 1.1 pooka tbl[i + 'a'] = 'a' + ((i + 13) % 26);
176 1.1 pooka for (i = 0; i < 26; i++)
177 1.1 pooka tbl[i + 'A'] = 'A' + ((i + 13) % 26);
178 1.1 pooka
179 1.4 pooka if (puffs_start(pu, pn_root, &svfsb) == -1)
180 1.1 pooka err(1, "puffs_start");
181 1.1 pooka
182 1.1 pooka if (puffs_mainloop(pu, lflags) == -1)
183 1.1 pooka err(1, "mainloop");
184 1.1 pooka
185 1.1 pooka return 0;
186 1.1 pooka }
187 1.1 pooka
188 1.1 pooka int
189 1.1 pooka rot13_node_readdir(struct puffs_cc *pcc, void *opc, struct dirent *dent,
190 1.3 pooka off_t *readoff, size_t *reslen, const struct puffs_cred *pcr,
191 1.3 pooka int *eofflag, off_t *cookies, size_t *ncookies)
192 1.1 pooka {
193 1.1 pooka struct dirent *dp;
194 1.1 pooka size_t rl;
195 1.1 pooka int rv;
196 1.1 pooka
197 1.1 pooka dp = dent;
198 1.1 pooka rl = *reslen;
199 1.1 pooka
200 1.3 pooka rv = puffs_null_node_readdir(pcc, opc, dent, readoff, reslen, pcr,
201 1.3 pooka eofflag, cookies, ncookies);
202 1.1 pooka if (rv)
203 1.1 pooka return rv;
204 1.1 pooka
205 1.1 pooka while (rl > *reslen) {
206 1.1 pooka dorot13((uint8_t *)dp->d_name, dp->d_namlen);
207 1.1 pooka rl -= _DIRENT_SIZE(dp);
208 1.1 pooka dp = _DIRENT_NEXT(dp);
209 1.1 pooka }
210 1.1 pooka
211 1.1 pooka return 0;
212 1.1 pooka }
213 1.1 pooka
214 1.1 pooka int
215 1.1 pooka rot13_node_read(struct puffs_cc *pcc, void *opc, uint8_t *buf, off_t offset,
216 1.1 pooka size_t *resid, const struct puffs_cred *pcr, int ioflag)
217 1.1 pooka {
218 1.1 pooka uint8_t *prebuf = buf;
219 1.1 pooka size_t preres = *resid;
220 1.1 pooka int rv;
221 1.1 pooka
222 1.1 pooka rv = puffs_null_node_read(pcc, opc, buf, offset, resid, pcr, ioflag);
223 1.1 pooka if (rv)
224 1.1 pooka return rv;
225 1.1 pooka
226 1.1 pooka dorot13(prebuf, preres - *resid);
227 1.1 pooka
228 1.1 pooka return rv;
229 1.1 pooka }
230 1.1 pooka
231 1.1 pooka int
232 1.1 pooka rot13_node_write(struct puffs_cc *pcc, void *opc, uint8_t *buf, off_t offset,
233 1.1 pooka size_t *resid, const struct puffs_cred *pcr, int ioflag)
234 1.1 pooka {
235 1.1 pooka
236 1.1 pooka dorot13(buf, *resid);
237 1.1 pooka return puffs_null_node_write(pcc, opc, buf, offset, resid, pcr, ioflag);
238 1.1 pooka }
239