-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathshfs.c
980 lines (871 loc) · 28.1 KB
/
shfs.c
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
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
/*
* Simple hash filesystem (SHFS)
*
* Authors: Simon Kuenzer <[email protected]>
*
*
* Copyright (c) 2013-2017, NEC Europe Ltd., NEC Corporation All rights reserved
*
* This software is available to you under a choice of one of two
* licenses. You may choose to be licensed under the terms of the GNU
* General Public License (GPL) Version 2, or the BSD license below:
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
*/
#include <target/sys.h>
#ifndef __KERNEL__
#include <stdint.h>
#include <errno.h>
#include "shfs_cache.h"
#else
int shfs_errno;
#endif
#include "shfs.h"
#include "shfs_check.h"
#include "shfs_defs.h"
#include "shfs_btable.h"
#ifdef SHFS_STATS
#include "shfs_stats_data.h"
#include "shfs_stats.h"
#endif
#ifdef SHFS_DEBUG
#define ENABLE_DEBUG
#endif
#include "debug.h"
#ifndef CACHELINE_SIZE
#define CACHELINE_SIZE 64
#endif
#if defined TRACE_BOOTTIME && CONFIG_AUTOMOUNT
#define TT_DECLARE(var) uint64_t (var) = 0
#define TT_START(var) do { (var) = target_now_ns(); } while(0)
#define TT_END(var) do { (var) = (target_now_ns() - (var)); } while(0)
TT_DECLARE(shfs_tt_vbdopen);
#else /* TRACE_BOOTTIME */
#define TT_DECLARE(var) while(0) {}
#define TT_START(var) while(0) {}
#define TT_END(var) while(0) {}
#endif
int shfs_mounted = 0;
unsigned int shfs_nb_open = 0;
sem_t shfs_mount_lock;
struct vol_info shfs_vol;
int init_shfs(void) {
init_SEMAPHORE(&shfs_mount_lock, 1);
return 0;
}
void exit_shfs(void) {
BUG_ON(shfs_mounted);
}
/**
* This function tries to open a blkdev and checks if it has a valid SHFS label
* On success, it returns the opened blkdev descriptor and the read disk chk0
* on *chk0
* On errors, a null pointer is returned
*
* Note: chk0 has to be a buffer of 4096 bytes and be aligned to 4096 bytes
*/
static struct blkdev *shfs_checkopen_blkdev(blkdev_id_t bd_id, void *chk0, int mode)
{
struct blkdev *bd;
#ifdef SHFS_DEBUG
char str_id[64];
#endif
sector_t rlen;
int ret;
TT_DECLARE(_tt_vbdopen);
#ifdef SHFS_DEBUG
blkdev_id_unparse(bd_id, str_id, sizeof(str_id));
#endif
TT_START(_tt_vbdopen);
bd = open_blkdev(bd_id, mode);
TT_END(_tt_vbdopen);
#if defined TRACE_BOOTTIME && CONFIG_AUTOMOUNT
shfs_tt_vbdopen += _tt_vbdopen;
#endif
if (!bd) {
printd("Could not open %s: %s\n", str_id, strerror(errno));
goto err_out;
}
if (blkdev_ssize(bd) > 4096 || blkdev_ssize(bd) < 512 ||
!POWER_OF_2(blkdev_ssize(bd))) {
/* incompatible device */
printd("Incompatible block size on block device %s\n", str_id);
goto err_close_bd;
}
/* read first chunk (chunksize considered as 4K) */
rlen = 4096 / blkdev_ssize(bd);
ret = blkdev_sync_read(bd, 0, rlen, chk0); /* FIXME: Use version that does not call schedule() */
if (ret < 0) {
printd("Could not read from block device %s: %d\n", str_id, ret);
errno = -ret;
goto err_close_bd;
}
/* Try to detect the SHFS disk label */
ret = shfs_detect_hdr0(chk0);
if (ret < 0) {
printd("Invalid or unsupported SHFS label detected on block device %s: %d\n", str_id, ret);
errno = -ret;
goto err_close_bd;
}
return bd;
err_close_bd:
close_blkdev(bd);
err_out:
return NULL;
}
/**
* This function iterates over bd_ids, tries to detect a SHFS label
* and does the low-level setup for mounting a volume
*/
static int load_vol_cconf(blkdev_id_t bd_id[], unsigned int count)
{
struct blkdev *bd;
struct vol_member detected_member[MAX_NB_TRY_BLKDEVS];
struct shfs_hdr_common *hdr_common;
unsigned int i;
uint8_t m;
unsigned int nb_detected_members;
uint64_t min_member_size;
int ret = 0;
sector_t rlen;
void *chk0;
int inuse;
#ifdef SHFS_DEBUG
char str_id[64];
#endif
if (count > MAX_NB_TRY_BLKDEVS) {
ret = -EINVAL;
goto err_out;
}
chk0 = target_malloc(4096, 4096);
if (!chk0) {
ret = -ENOMEM;
goto err_out;
}
/* Iterate over block devices and try to find those with a valid SHFS disk label */
nb_detected_members = 0;
for (i = 0; i < count; i++) {
#ifdef SHFS_DEBUG
blkdev_id_unparse(bd_id[i], str_id, sizeof(str_id));
printd("Search for SHFS label on device %s...\n", str_id);
#endif
bd = shfs_checkopen_blkdev(bd_id[i], chk0, O_RDONLY);
if (!bd) {
continue; /* try next device */
}
#ifdef SHFS_DEBUG
printd("Supported SHFS label detected on %s\n", str_id);
#endif
/* chk0 now contains the first chunk read from disk */
hdr_common = (void *)((uint8_t *) chk0 + BOOT_AREA_LENGTH);
detected_member[nb_detected_members].bd = bd;
uuid_copy(detected_member[nb_detected_members].uuid, hdr_common->member_uuid);
nb_detected_members++;
}
if (nb_detected_members == 0) {
ret = -ENODEV;
goto err_free_chk0;
}
/* Load label from first detected member */
#ifdef SHFS_DEBUG
blkdev_id_unparse(bd_id[i], str_id, sizeof(str_id));
printd("Load SHFS label from %s...\n", str_id);
#endif
rlen = 4096 / blkdev_ssize(detected_member[0].bd);
ret = blkdev_sync_read(detected_member[0].bd, 0, rlen, chk0);
if (ret < 0)
goto err_close_bds;
hdr_common = (void *)((uint8_t *) chk0 + BOOT_AREA_LENGTH);
memcpy(shfs_vol.uuid, hdr_common->vol_uuid, 16);
memcpy(shfs_vol.volname, hdr_common->vol_name, 16);
shfs_vol.volname[16] = '\0'; /* ensure nullterminated volume name */
shfs_vol.ts_creation = hdr_common->vol_ts_creation;
shfs_vol.stripesize = hdr_common->member_stripesize;
shfs_vol.stripemode = hdr_common->member_stripemode;
#if defined CONFIG_SELECT_POLL && defined CAN_POLL_BLKDEV
shfs_vol.members_maxfd = blkdev_get_fd(detected_member[0].bd);
#endif
if (shfs_vol.stripemode != SHFS_SM_COMBINED &&
shfs_vol.stripemode != SHFS_SM_INDEPENDENT) {
printd("Stripe mode 0x%x is not supported\n", shfs_vol.stripemode);
ret = -ENOTSUP;
goto err_close_bds;
}
shfs_vol.chunksize = SHFS_CHUNKSIZE(hdr_common);
shfs_vol.volsize = hdr_common->vol_size;
/* Find and add members to the volume */
printd("Searching for members of volume '%s'...\n", shfs_vol.volname);
shfs_vol.nb_members = 0;
for (i = 0; i < hdr_common->member_count; i++) {
for (m = 0; m < nb_detected_members; ++m) {
if (uuid_compare(hdr_common->member[i].uuid, detected_member[m].uuid) == 0) {
/* found device */
#ifdef SHFS_DEBUG
blkdev_id_unparse(blkdev_id(detected_member[m].bd), str_id, sizeof(str_id));
printd(" Member %u/%"PRIu8" is block device %s\n",
i + 1, hdr_common->member_count, str_id);
#endif
shfs_vol.member[shfs_vol.nb_members].bd = detected_member[m].bd;
uuid_copy(shfs_vol.member[shfs_vol.nb_members].uuid, detected_member[m].uuid);
#if defined CONFIG_SELECT_POLL && defined CAN_POLL_BLKDEV
shfs_vol.members_maxfd = max(shfs_vol.members_maxfd,
blkdev_get_fd(detected_member[m].bd));
#endif
shfs_vol.nb_members++;
break;
}
}
}
if (shfs_vol.nb_members != hdr_common->member_count) {
printd("Could not find all members for volume '%s'\n",
shfs_vol.volname);
ret = -ENOENT;
goto err_close_bds;
}
/* chunk and stripe size -> retrieve a device sector factor for each device and
* also the alignment requirements for io buffers */
if (shfs_vol.stripesize > 32768 || shfs_vol.stripesize < 4096 ||
!POWER_OF_2(shfs_vol.stripesize)) {
printd("Stripe size invalid on volume '%s'\n",
shfs_vol.volname);
ret = -ENOENT;
goto err_close_bds;
}
shfs_vol.ioalign = 0;
for (i = 0; i < shfs_vol.nb_members; ++i) {
/* ioalign for all chunk buffers
* Note: Since the chunk buffers are accessed by all devices,
* the final ioalign value has to a multiple of all
* device-dependent ioaligns */
if (blkdev_ioalign(shfs_vol.member[i].bd) > shfs_vol.ioalign) {
ASSERT(!shfs_vol.ioalign ||
blkdev_ioalign(shfs_vol.member[i].bd) % shfs_vol.ioalign == 0);
shfs_vol.ioalign = blkdev_ioalign(shfs_vol.member[i].bd);
} else {
ASSERT(shfs_vol.ioalign % blkdev_ioalign(shfs_vol.member[i].bd) == 0);
}
/* device dependent 'stripe-to-sector' factor */
shfs_vol.member[i].sfactor = shfs_vol.stripesize / blkdev_ssize(shfs_vol.member[i].bd);
if (shfs_vol.member[i].sfactor == 0) {
printd("Stripe size invalid on volume '%s'\n",
shfs_vol.volname);
ret = -ENOENT;
goto err_close_bds;
}
}
/* calculate and check volume size */
if (shfs_vol.stripemode == SHFS_SM_COMBINED)
min_member_size = (shfs_vol.volsize + 1) * (uint64_t) shfs_vol.stripesize;
else /* SHFS_SM_INTERLEAVED */
min_member_size = ((shfs_vol.volsize + 1) / shfs_vol.nb_members) * (uint64_t) shfs_vol.stripesize;
for (i = 0; i < shfs_vol.nb_members; ++i) {
if (blkdev_size(shfs_vol.member[i].bd) < min_member_size) {
printd("Member %u of volume '%s' is too small\n",
i, shfs_vol.volname);
ret = -ENOENT;
goto err_close_bds;
}
}
/* clean-up: close non-used devices */
for (m = 0; m < nb_detected_members; ++m) {
inuse = 0;
for (i = 0; i < shfs_vol.nb_members; ++i) {
if (detected_member[m].bd == shfs_vol.member[i].bd) {
inuse = 1;
break;
}
}
if (!inuse)
close_blkdev(detected_member[m].bd);
}
target_free(chk0);
return 0;
err_close_bds:
for (m = 0; m < nb_detected_members; ++m)
close_blkdev(detected_member[m].bd);
err_free_chk0:
target_free(chk0);
err_out:
return ret;
}
/**
* This function loads the hash configuration from chunk 1
* (as defined in SHFS)
* This function can only be called, after load_vol_cconf
* established successfully the low-level setup of a volume
* (required for chunk I/O)
*/
static int load_vol_hconf(void)
{
struct shfs_hdr_config *hdr_config;
void *chk1;
int ret;
chk1 = target_malloc(4096, shfs_vol.chunksize);
if (!chk1) {
ret = -ENOMEM;
goto out;
}
printd("Loading SHFS configuration chunk...\n");
ret = shfs_read_chunk_nosched(1, 1, chk1);
if (ret < 0)
goto out_free_chk1;
hdr_config = chk1;
shfs_vol.htable_ref = hdr_config->htable_ref;
shfs_vol.htable_bak_ref = hdr_config->htable_bak_ref;
shfs_vol.htable_nb_buckets = hdr_config->htable_bucket_count;
shfs_vol.htable_nb_entries_per_bucket = hdr_config->htable_entries_per_bucket;
shfs_vol.htable_nb_entries = SHFS_HTABLE_NB_ENTRIES(hdr_config);
shfs_vol.htable_nb_entries_per_chunk = SHFS_HENTRIES_PER_CHUNK(shfs_vol.chunksize);
shfs_vol.htable_len = SHFS_HTABLE_SIZE_CHUNKS(hdr_config, shfs_vol.chunksize);
shfs_vol.hlen = hdr_config->hlen;
ret = 0;
/* brief configuration check */
if (shfs_vol.htable_len == 0) {
printd("Malformed SHFS configuration\n");
ret = -ENOENT;
goto out_free_chk1;
}
out_free_chk1:
target_free(chk1);
out:
return ret;
}
/**
* This function loads the hash table from the block device into memory
* Note: load_vol_hconf() and local_vol_cconf() has to called before
*/
struct _load_vol_htable_aiot {
int done;
chk_t left;
int ret;
};
static void _load_vol_htable_cb(SHFS_AIO_TOKEN *t, void *cookie, void *argp)
{
struct _load_vol_htable_aiot *aiot = (struct _load_vol_htable_aiot *) cookie;
register int ioret;
printd("*** AIO HTABLE CB (ret = %d / left = %"PRIu64") ***\n", aiot->ret, aiot->left - 1);
BUG_ON(aiot->left == 0); /* This happens most likely when more requests are
* sent to device than it can handle -> check MAX_REQUESTS
* in blkdev.h */
ioret = shfs_aio_finalize(t);
if (unlikely(ioret < 0))
aiot->ret = ioret;
--aiot->left;
if (unlikely(aiot->left == 0))
aiot->done = 1;
}
static int load_vol_htable(void)
{
struct _load_vol_htable_aiot aiot;
SHFS_AIO_TOKEN *aioret;
struct shfs_hentry *hentry;
struct shfs_bentry *bentry;
void *chk_buf;
unsigned int i;
chk_t c;
int ret;
printd("Allocating chunk cache reference table (size: %lu B)...\n",
sizeof(void *) * shfs_vol.htable_len);
shfs_vol.htable_chunk_cache = target_malloc(CACHELINE_SIZE, sizeof(void *) * shfs_vol.htable_len);
if (!shfs_vol.htable_chunk_cache) {
ret = -ENOMEM;
goto err_out;
}
memset(shfs_vol.htable_chunk_cache, 0, sizeof(void *) * shfs_vol.htable_len);
/* read hash table from device */
aiot.done = 0;
aiot.left = shfs_vol.htable_len;
aiot.ret = 0;
for (c = 0; c < shfs_vol.htable_len; ++c) {
/* allocate buffer and register it to htable chunk cache */
printd("Allocate buffer for chunk %"PRIchk" of htable (size: %lu B, align: %"PRIu32")\n",
c, shfs_vol.chunksize, shfs_vol.ioalign);
chk_buf = target_malloc(shfs_vol.ioalign, shfs_vol.chunksize);
if (!chk_buf) {
printd("Could not alloc chunk %"PRIchk"\n", c);
ret = -ENOMEM;
goto err_free_chunkcache;
}
shfs_vol.htable_chunk_cache[c] = chk_buf;
repeat_aio:
printd("Setup async read for chunk %"PRIchk"\n", c);
aioret = shfs_aread_chunk(shfs_vol.htable_ref + c, 1, chk_buf,
_load_vol_htable_cb, &aiot, NULL);
if (!aioret && (errno == EAGAIN || errno == EBUSY)) {
printd("Device is busy: Retrying...\n");
shfs_aio_submit();
shfs_poll_blkdevs();
goto repeat_aio;
}
if (!aioret) {
printd("Could not setup async read: %s\n", strerror(errno));
aiot.left -= (shfs_vol.htable_len - c);
goto err_cancel_aio;
}
}
shfs_aio_submit();
/* allocate bucket table */
printd("Allocating btable...\n");
shfs_vol.bt = shfs_alloc_btable(shfs_vol.htable_nb_buckets,
shfs_vol.htable_nb_entries_per_bucket,
shfs_vol.hlen);
if (!shfs_vol.bt) {
ret = -ENOMEM;
goto err_free_chunkcache;
}
/* wait for I/O completion */
printd("Waiting for I/O completion...\n");
while (!aiot.done)
shfs_poll_blkdevs();
if (aiot.ret < 0) {
printd("There was an I/O error: Aborting...\n");
ret = -EIO;
goto err_free_btable;
}
/* feed bucket table */
shfs_vol.def_bentry = NULL;
printd("Feeding hash table...\n");
for (i = 0; i < shfs_vol.htable_nb_entries; ++i) {
c = SHFS_HTABLE_CHUNK_NO(i, shfs_vol.htable_nb_entries_per_chunk);
chk_buf = shfs_vol.htable_chunk_cache[c];
hentry = (struct shfs_hentry *)((uint8_t *) chk_buf
+ SHFS_HTABLE_ENTRY_OFFSET(i, shfs_vol.htable_nb_entries_per_chunk));
bentry = shfs_btable_feed(shfs_vol.bt, i, hentry->hash);
bentry->hentry = hentry;
bentry->hentry_htchunk = c;
bentry->hentry_htoffset = SHFS_HTABLE_ENTRY_OFFSET(i, shfs_vol.htable_nb_entries_per_chunk);
bentry->refcount = 0;
bentry->update = 0;
#ifdef __KERNEL__
bentry->ino = i + LINUX_FIRST_INO_N;
#endif
init_SEMAPHORE(&bentry->updatelock, 1);
#ifdef SHFS_STATS
memset(&bentry->hstats, 0, sizeof(bentry->hstats));
#endif
if (SHFS_HENTRY_ISDEFAULT(hentry))
shfs_vol.def_bentry = bentry;
}
return 0;
err_cancel_aio:
if (aiot.left) {
while (!trydown(&aiot.done))
shfs_poll_blkdevs();
}
ret = -EIO;
goto err_free_chunkcache;
err_free_btable:
shfs_free_btable(shfs_vol.bt);
err_free_chunkcache:
for (i = 0; i < shfs_vol.htable_len; ++i) {
if (shfs_vol.htable_chunk_cache[i])
target_free(shfs_vol.htable_chunk_cache[i]);
}
target_free(shfs_vol.htable_chunk_cache);
err_out:
return ret;
}
#ifndef __KERNEL__
static void _aiotoken_pool_objinit(struct mempool_obj *, void *);
#endif
/**
* Mount a SHFS volume
* The volume is searched on the given list of block devices
*/
int mount_shfs(blkdev_id_t bd_id[], unsigned int count)
{
unsigned int i;
int ret;
down(&shfs_mount_lock);
if (count == 0) {
ret = -EINVAL;
goto err_out;
}
if (shfs_mounted) {
ret = -EALREADY;
goto err_out;
}
shfs_mounted = 0;
/* load common volume information and open devices */
printd("Loading common volume information...\n");
ret = load_vol_cconf(bd_id, count);
if (ret < 0)
goto err_out;
/* a memory pool required for async I/O requests (even on cache) */
shfs_vol.aiotoken_pool = alloc_mempool(NB_AIOTOKEN, sizeof(struct _shfs_aio_token),
0, 0, 0, _aiotoken_pool_objinit, NULL, 0);
if (!shfs_vol.aiotoken_pool)
goto err_close_members;
shfs_mounted = 1; /* required by next function calls */
/* load hash conf (uses shfs_sync_read_chunk) */
printd("Loading volume configuration...\n");
ret = load_vol_hconf();
if (ret < 0)
goto err_free_aiotoken_pool;
/* load htable (uses shfs_sync_read_chunk)
* This function also allocates htable_chunk_cache,
* htable_chunk_cache_state and btable */
printd("Loading volume hash table...\n");
ret = load_vol_htable();
if (ret < 0)
goto err_close_members;
printd("Allocating remount chunk buffer...\n");
shfs_vol.remount_chunk_buffer = target_malloc(shfs_vol.ioalign, shfs_vol.chunksize);
if (!shfs_vol.remount_chunk_buffer)
goto err_free_htable;
/* chunk buffer cache for I/O */
printd("Allocating chunk cache...\n");
ret = shfs_alloc_cache();
if (ret < 0)
goto err_free_remount_buffer;
#ifdef SHFS_STATS
printd("Initializing statistics...\n");
ret = shfs_init_mstats(shfs_vol.htable_nb_buckets,
shfs_vol.htable_nb_entries_per_bucket,
shfs_vol.hlen);
if (!ret < 0) {
shfs_mounted = 0;
goto err_free_chunkcache;
}
#endif
shfs_nb_open = 0;
up(&shfs_mount_lock);
printd("SHFS volume mounted\n");
return 0;
/* make compiller happy */
goto err_free_chunkcache;
err_free_chunkcache:
shfs_free_cache();
err_free_remount_buffer:
target_free(shfs_vol.remount_chunk_buffer);
err_free_htable:
for (i = 0; i < shfs_vol.htable_len; ++i) {
if (shfs_vol.htable_chunk_cache[i])
target_free(shfs_vol.htable_chunk_cache[i]);
}
target_free(shfs_vol.htable_chunk_cache);
shfs_free_btable(shfs_vol.bt);
err_free_aiotoken_pool:
free_mempool(shfs_vol.aiotoken_pool);
err_close_members:
for(i = 0; i < shfs_vol.nb_members; ++i)
close_blkdev(shfs_vol.member[i].bd);
shfs_mounted = 0;
err_out:
up(&shfs_mount_lock);
return ret;
}
/**
* Unmounts a previously mounted SHFS volume
* Note: Because semaphores are used to sync with opened files,
* when force is enabled, this function has to be called
* from a context that is different from the one of the main loop
*/
int umount_shfs(int force) {
unsigned int i;
down(&shfs_mount_lock);
if (shfs_mounted) {
#ifndef __KERNEL__
if (shfs_nb_open ||
mempool_free_count(shfs_vol.aiotoken_pool) < MAX_REQUESTS ||
shfs_cache_ref_count()) {
struct htable_el *el;
/* there are still open files and/or async I/O is happening */
printd("Could not umount: SHFS is busy:\n");
printd(" Open files: %u\n",
shfs_nb_open);
printd(" Infly AIO tokens: %u\n",
MAX_REQUESTS - mempool_free_count(shfs_vol.aiotoken_pool));
printd(" Referenced chunk buffers: %"PRIu64"\n",
shfs_cache_ref_count());
if (!force) {
up(&shfs_mount_lock);
return -EBUSY;
}
/* lock entries */
foreach_htable_el(shfs_vol.bt, el) {
struct shfs_bentry *bentry = el->private;
bentry->update = 1; /* forbid further open() */
down(&bentry->updatelock); /* wait until file is closed */
}
}
shfs_free_cache();
#endif
shfs_mounted = 0;
target_free(shfs_vol.remount_chunk_buffer);
for (i = 0; i < shfs_vol.htable_len; ++i) {
if (shfs_vol.htable_chunk_cache[i])
target_free(shfs_vol.htable_chunk_cache[i]);
}
target_free(shfs_vol.htable_chunk_cache);
shfs_free_btable(shfs_vol.bt);
free_mempool(shfs_vol.aiotoken_pool);
for(i = 0; i < shfs_vol.nb_members; ++i)
close_blkdev(shfs_vol.member[i].bd); /* might call schedule() */
shfs_vol.nb_members = 0;
#ifdef SHFS_STATS
shfs_free_mstats();
#endif
}
up(&shfs_mount_lock);
return 0;
}
/**
* This function re-reads the hash table from the device
* Since semaphores are used to sync with opened files,
* this function has to be called from a context that
* is different from the one of the main loop
*/
static int reload_vol_htable(void) {
#ifdef SHFS_STATS
struct shfs_el_stats *el_stats;
#endif
struct shfs_bentry *bentry;
struct shfs_hentry *chentry;
struct shfs_hentry *nhentry;
void *cchk_buf;
void *nchk_buf = shfs_vol.remount_chunk_buffer;
int chash_is_zero, nhash_is_zero;
register chk_t c;
register unsigned int e;
int ret = 0;
printd("Re-reading hash table...\n");
for (c = 0; c < shfs_vol.htable_len; ++c) {
/* read chunk from disk */
ret = shfs_read_chunk(shfs_vol.htable_ref + c, 1, nchk_buf); /* calls schedule() */
if (ret < 0) {
ret = -EIO;
goto out;
}
cchk_buf = shfs_vol.htable_chunk_cache[c];
/* compare entries */
for (e = 0; e < shfs_vol.htable_nb_entries_per_chunk; ++e) {
chentry = (struct shfs_hentry *)((uint8_t *) cchk_buf
+ SHFS_HTABLE_ENTRY_OFFSET(e, shfs_vol.htable_nb_entries_per_chunk));
nhentry = (struct shfs_hentry *)((uint8_t *) nchk_buf
+ SHFS_HTABLE_ENTRY_OFFSET(e, shfs_vol.htable_nb_entries_per_chunk));
if (hash_compare(chentry->hash, nhentry->hash, shfs_vol.hlen)) {
chash_is_zero = hash_is_zero(chentry->hash, shfs_vol.hlen);
nhash_is_zero = hash_is_zero(nhentry->hash, shfs_vol.hlen);
if (!chash_is_zero || !nhash_is_zero) { /* process only if at least one hash
* digest is non-zero */
printd("Chunk %"PRIchk", entry %u has been updated\n", c ,e);
/* Update hash of entry
* Note: Any open file should not be affected, because
* there is no hash table lookup needed again
* The meta data is updated after all handles were closed
* Note: Since we lock the file in the next step,
* upcoming open of this entry will only be successful
* when the update has been finished */
bentry = shfs_btable_feed(shfs_vol.bt,
(c * shfs_vol.htable_nb_entries_per_chunk) + e,
nhentry->hash);
/* lock entry */
bentry->update = 1; /* forbid further open() */
down(&bentry->updatelock); /* wait until files is closed */
#ifdef SHFS_STATS
if (!chash_is_zero) {
/* move current stats to miss table */
el_stats = shfs_stats_from_mstats(chentry->hash);
if (likely(el_stats != NULL))
memcpy(el_stats, &bentry->hstats, sizeof(*el_stats));
/* reset stats of element */
memset(&bentry->hstats, 0, sizeof(*el_stats));
} else {
/* load stats from miss table */
el_stats = shfs_stats_from_mstats(nhentry->hash);
if (likely(el_stats != NULL))
memcpy(&bentry->hstats, el_stats, sizeof(*el_stats));
else
memset(&bentry->hstats, 0, sizeof(*el_stats));
/* delete entry from miss stats */
shfs_stats_mstats_drop(nhentry->hash);
}
#endif
memcpy(chentry, nhentry, sizeof(*chentry));
shfs_flush_cache();
/* unlock entry */
up(&bentry->updatelock);
bentry->update = 0;
/* update default entry reference */
if (shfs_vol.def_bentry == bentry &&
!SHFS_HENTRY_ISDEFAULT(nhentry))
shfs_vol.def_bentry = NULL;
else if (SHFS_HENTRY_ISDEFAULT(nhentry))
shfs_vol.def_bentry = bentry;
}
} else {
/* in this case, at most the file location has been moved
* or the contents has been changed
*
* Note: This is usually a bad thing but happens
* if the tools were misused
* Note: Since the hash digest did not change,
* the stats keep the same */
bentry = shfs_btable_feed(shfs_vol.bt,
(c * shfs_vol.htable_nb_entries_per_chunk) + e,
nhentry->hash);
/* lock entry */
bentry->update = 1; /* forbid further open() */
down(&bentry->updatelock); /* wait until this file is closed */
memcpy(chentry, nhentry, sizeof(*chentry));
shfs_flush_cache(); /* to ensure re-reading this file */
/* unlock entry */
up(&bentry->updatelock);
bentry->update = 0;
/* update default entry reference */
if (shfs_vol.def_bentry == bentry &&
!SHFS_HENTRY_ISDEFAULT(nhentry))
shfs_vol.def_bentry = NULL;
else if (SHFS_HENTRY_ISDEFAULT(nhentry))
shfs_vol.def_bentry = bentry;
}
}
}
out:
return ret;
}
/**
* This function re-reads the hash table from the device
* Since semaphores are used to sync with opened files,
* this function has to be called from a context that
* is different from the one of the main loop
*/
int remount_shfs(void) {
int ret;
down(&shfs_mount_lock);
if (!shfs_mounted) {
ret = -ENODEV;
goto out;
}
/* TODO: Re-read chunk0 and check if volume UUID still matches */
ret = reload_vol_htable();
out:
up(&shfs_mount_lock);
return ret;
}
/*
* Note: Async I/O token data access is atomic since none of these functions are
* interrupted or can yield the CPU. Even blkfront calls the callbacks outside
* of the interrupt context via blkdev_poll_req() and there is only the
* cooperative scheduler...
*/
#ifndef __KERNEL__
static void _aiotoken_pool_objinit(struct mempool_obj *t_obj, void *argp)
{
SHFS_AIO_TOKEN *t;
t = t_obj->data;
t->p_obj = t_obj;
t->ret = 0;
t->infly = 0;
t->cb = NULL;
t->cb_argp = NULL;
t->cb_cookie = NULL;
}
#endif
static void _shfs_aio_cb(int ret, void *argp) {
SHFS_AIO_TOKEN *t = argp;
if (unlikely(ret < 0))
t->ret = ret;
--t->infly;
if (t->infly == 0) {
/* call user's callback */
if (t->cb)
t->cb(t, t->cb_cookie, t->cb_argp);
}
}
SHFS_AIO_TOKEN *shfs_aio_chunk(chk_t start, chk_t len, int write, void *buffer,
shfs_aiocb_t *cb, void *cb_cookie, void *cb_argp)
{
int ret;
uint64_t num_req_per_member;
sector_t start_sec;
unsigned int m;
uint8_t *ptr = buffer;
SHFS_AIO_TOKEN *t;
strp_t start_s;
strp_t end_s;
strp_t strp;
if (!shfs_mounted) {
errno = ENODEV;
goto err_out;
}
switch (shfs_vol.stripemode) {
case SHFS_SM_COMBINED:
start_s = (strp_t) start * (strp_t) shfs_vol.nb_members;
end_s = (strp_t) (start + len) * (strp_t) shfs_vol.nb_members;
break;
case SHFS_SM_INDEPENDENT:
default:
start_s = (strp_t) start + (strp_t) (shfs_vol.nb_members - 1);
end_s = (strp_t) (start_s + len);
break;
}
num_req_per_member = (end_s - start_s) / shfs_vol.nb_members;
/* check if each member has enough request objects available for this operation */
for (m = 0; m < shfs_vol.nb_members; ++m) {
if (blkdev_avail_req(shfs_vol.member[m].bd) < num_req_per_member) {
errno = EAGAIN;
goto err_out;
}
}
/* pick token */
t = shfs_aio_pick_token();
if (!t) {
errno = EAGAIN;
goto err_out;
}
t->cb = cb;
t->cb_argp = cb_argp;
t->cb_cookie = cb_cookie;
/* setup requests */
for (strp = start_s; strp < end_s; ++strp) {
/* TODO: Try using shifts and masks
* instead of multiplies, mods and divs */
m = strp % shfs_vol.nb_members;
start_sec = (strp / shfs_vol.nb_members) * shfs_vol.member[m].sfactor;
printd("Request: member=%u, start=%"PRIsctr"s, len=%"PRIsctr"s, dataptr=@%p\n",
m, start_sec, shfs_vol.member[m].sfactor, ptr);
ret = blkdev_async_io(shfs_vol.member[m].bd, start_sec, shfs_vol.member[m].sfactor,
write, ptr, _shfs_aio_cb, t);
if (unlikely(ret < 0)) {
t->cb = NULL; /* erase callback */
printd("Error while setting up async I/O request for member %u: %d. "
"Cancelling request...\n", m, ret);
shfs_aio_wait(t);
errno = -ret;
goto err_free_token;
}
++t->infly;
ptr += shfs_vol.stripesize;
}
return t;
err_free_token:
shfs_aio_put_token(t);
err_out:
return NULL;
}