hc
2024-03-26 e0728245c89800c2038c23308f2d88969d5b41c8
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
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (c) 2013
 * Phillip Lougher <phillip@squashfs.org.uk>
 */
 
#include <linux/types.h>
#include <linux/slab.h>
#include <linux/percpu.h>
#include <linux/buffer_head.h>
#include <linux/local_lock.h>
 
#include "squashfs_fs.h"
#include "squashfs_fs_sb.h"
#include "decompressor.h"
#include "squashfs.h"
 
/*
 * This file implements multi-threaded decompression using percpu
 * variables, one thread per cpu core.
 */
 
struct squashfs_stream {
   void            *stream;
   local_lock_t    lock;
};
 
void *squashfs_decompressor_create(struct squashfs_sb_info *msblk,
                       void *comp_opts)
{
   struct squashfs_stream *stream;
   struct squashfs_stream __percpu *percpu;
   int err, cpu;
 
   percpu = alloc_percpu(struct squashfs_stream);
   if (percpu == NULL)
       return ERR_PTR(-ENOMEM);
 
   for_each_possible_cpu(cpu) {
       stream = per_cpu_ptr(percpu, cpu);
       stream->stream = msblk->decompressor->init(msblk, comp_opts);
       if (IS_ERR(stream->stream)) {
           err = PTR_ERR(stream->stream);
           goto out;
       }
       local_lock_init(&stream->lock);
   }
 
   kfree(comp_opts);
   return (__force void *) percpu;
 
out:
   for_each_possible_cpu(cpu) {
       stream = per_cpu_ptr(percpu, cpu);
       if (!IS_ERR_OR_NULL(stream->stream))
           msblk->decompressor->free(stream->stream);
   }
   free_percpu(percpu);
   return ERR_PTR(err);
}
 
void squashfs_decompressor_destroy(struct squashfs_sb_info *msblk)
{
   struct squashfs_stream __percpu *percpu =
           (struct squashfs_stream __percpu *) msblk->stream;
   struct squashfs_stream *stream;
   int cpu;
 
   if (msblk->stream) {
       for_each_possible_cpu(cpu) {
           stream = per_cpu_ptr(percpu, cpu);
           msblk->decompressor->free(stream->stream);
       }
       free_percpu(percpu);
   }
}
 
int squashfs_decompress(struct squashfs_sb_info *msblk, struct bio *bio,
   int offset, int length, struct squashfs_page_actor *output)
{
   struct squashfs_stream *stream;
   int res;
 
   local_lock(&msblk->stream->lock);
   stream = this_cpu_ptr(msblk->stream);
 
   res = msblk->decompressor->decompress(msblk, stream->stream, bio,
                         offset, length, output);
 
   local_unlock(&msblk->stream->lock);
 
   if (res < 0)
       ERROR("%s decompression failed, data probably corrupt\n",
           msblk->decompressor->name);
 
   return res;
}
 
int squashfs_max_decompressors(void)
{
   return num_possible_cpus();
}