Coverage Report

Created: 2025-11-14 07:32

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/render_pipeline/render_pipeline.cc
Line
Count
Source
1
// Copyright (c) the JPEG XL Project Authors. All rights reserved.
2
//
3
// Use of this source code is governed by a BSD-style
4
// license that can be found in the LICENSE file.
5
6
#include "lib/jxl/render_pipeline/render_pipeline.h"
7
8
#include <cstddef>
9
#include <memory>
10
#include <utility>
11
#include <vector>
12
13
#include "lib/jxl/base/common.h"
14
#include "lib/jxl/base/rect.h"
15
#include "lib/jxl/base/sanitizers.h"
16
#include "lib/jxl/base/status.h"
17
#include "lib/jxl/frame_dimensions.h"
18
#include "lib/jxl/image.h"
19
#include "lib/jxl/render_pipeline/low_memory_render_pipeline.h"
20
#include "lib/jxl/render_pipeline/render_pipeline_stage.h"
21
#include "lib/jxl/render_pipeline/simple_render_pipeline.h"
22
23
namespace jxl {
24
25
Status RenderPipeline::Builder::AddStage(
26
285k
    std::unique_ptr<RenderPipelineStage> stage) {
27
285k
  if (!stage) return JXL_FAILURE("internal: no stage to add");
28
285k
  stages_.push_back(std::move(stage));
29
285k
  return true;
30
285k
}
31
32
StatusOr<std::unique_ptr<RenderPipeline>> RenderPipeline::Builder::Finalize(
33
43.2k
    FrameDimensions frame_dimensions) && {
34
  // Check that the last stage is not a kInOut stage for any channel, and that
35
  // there is at least one stage.
36
43.2k
  JXL_ENSURE(!stages_.empty());
37
203k
  for (size_t c = 0; c < num_c_; c++) {
38
160k
    JXL_ENSURE(stages_.back()->GetChannelMode(c) !=
39
160k
               RenderPipelineChannelMode::kInOut);
40
160k
  }
41
42
43.2k
  std::unique_ptr<RenderPipeline> res;
43
43.2k
  if (use_simple_implementation_) {
44
0
    res = jxl::make_unique<SimpleRenderPipeline>(memory_manager_);
45
43.2k
  } else {
46
43.2k
    res = jxl::make_unique<LowMemoryRenderPipeline>(memory_manager_);
47
43.2k
  }
48
49
43.2k
  res->padding_.resize(stages_.size());
50
328k
  for (size_t i = stages_.size(); i-- > 0;) {
51
285k
    const auto& stage = stages_[i];
52
285k
    res->padding_[i].resize(num_c_);
53
285k
    if (i + 1 == stages_.size()) {
54
43.2k
      continue;
55
43.2k
    }
56
1.29M
    for (size_t c = 0; c < num_c_; c++) {
57
1.05M
      if (stage->GetChannelMode(c) == RenderPipelineChannelMode::kInOut) {
58
285k
        res->padding_[i][c].first = DivCeil(res->padding_[i + 1][c].first,
59
285k
                                            1 << stage->settings_.shift_x) +
60
285k
                                    stage->settings_.border_x;
61
285k
        res->padding_[i][c].second = DivCeil(res->padding_[i + 1][c].second,
62
285k
                                             1 << stage->settings_.shift_y) +
63
285k
                                     stage->settings_.border_y;
64
769k
      } else {
65
769k
        res->padding_[i][c] = res->padding_[i + 1][c];
66
769k
      }
67
1.05M
    }
68
241k
  }
69
70
43.2k
  res->frame_dimensions_ = frame_dimensions;
71
43.2k
  res->group_completed_passes_.resize(frame_dimensions.num_groups);
72
43.2k
  res->channel_shifts_.resize(stages_.size());
73
43.2k
  res->channel_shifts_[0].resize(num_c_);
74
285k
  for (size_t i = 1; i < stages_.size(); i++) {
75
241k
    auto& stage = stages_[i - 1];
76
1.29M
    for (size_t c = 0; c < num_c_; c++) {
77
1.05M
      if (stage->GetChannelMode(c) == RenderPipelineChannelMode::kInOut) {
78
285k
        res->channel_shifts_[0][c].first += stage->settings_.shift_x;
79
285k
        res->channel_shifts_[0][c].second += stage->settings_.shift_y;
80
285k
      }
81
1.05M
    }
82
241k
  }
83
285k
  for (size_t i = 1; i < stages_.size(); i++) {
84
241k
    auto& stage = stages_[i - 1];
85
241k
    res->channel_shifts_[i].resize(num_c_);
86
1.29M
    for (size_t c = 0; c < num_c_; c++) {
87
1.05M
      if (stage->GetChannelMode(c) == RenderPipelineChannelMode::kInOut) {
88
285k
        res->channel_shifts_[i][c].first =
89
285k
            res->channel_shifts_[i - 1][c].first - stage->settings_.shift_x;
90
285k
        res->channel_shifts_[i][c].second =
91
285k
            res->channel_shifts_[i - 1][c].second - stage->settings_.shift_y;
92
769k
      } else {
93
769k
        res->channel_shifts_[i][c].first = res->channel_shifts_[i - 1][c].first;
94
769k
        res->channel_shifts_[i][c].second =
95
769k
            res->channel_shifts_[i - 1][c].second;
96
769k
      }
97
1.05M
    }
98
241k
  }
99
43.2k
  res->stages_ = std::move(stages_);
100
43.2k
  JXL_RETURN_IF_ERROR(res->Init());
101
43.2k
  return res;
102
43.2k
}
103
104
RenderPipelineInput RenderPipeline::GetInputBuffers(size_t group_id,
105
96.2k
                                                    size_t thread_id) {
106
96.2k
  RenderPipelineInput ret;
107
96.2k
  JXL_DASSERT(group_id < group_completed_passes_.size());
108
96.2k
  ret.group_id_ = group_id;
109
96.2k
  ret.thread_id_ = thread_id;
110
96.2k
  ret.pipeline_ = this;
111
96.2k
  ret.buffers_ = PrepareBuffers(group_id, thread_id);
112
96.2k
  return ret;
113
96.2k
}
114
115
Status RenderPipeline::InputReady(
116
    size_t group_id, size_t thread_id,
117
58.8k
    const std::vector<std::pair<ImageF*, Rect>>& buffers) {
118
58.8k
  JXL_ENSURE(group_id < group_completed_passes_.size());
119
58.8k
  group_completed_passes_[group_id]++;
120
268k
  for (size_t i = 0; i < buffers.size(); ++i) {
121
209k
    (void)i;
122
209k
    JXL_CHECK_PLANE_INITIALIZED(*buffers[i].first, buffers[i].second, i);
123
209k
  }
124
125
58.8k
  JXL_RETURN_IF_ERROR(ProcessBuffers(group_id, thread_id));
126
58.8k
  return true;
127
58.8k
}
128
129
71.2k
Status RenderPipeline::PrepareForThreads(size_t num, bool use_group_ids) {
130
498k
  for (const auto& stage : stages_) {
131
498k
    JXL_RETURN_IF_ERROR(stage->PrepareForThreads(num));
132
498k
  }
133
71.2k
  JXL_RETURN_IF_ERROR(PrepareForThreadsInternal(num, use_group_ids));
134
71.2k
  return true;
135
71.2k
}
136
137
58.8k
Status RenderPipelineInput::Done() {
138
58.8k
  JXL_ENSURE(pipeline_);
139
58.8k
  JXL_RETURN_IF_ERROR(pipeline_->InputReady(group_id_, thread_id_, buffers_));
140
58.8k
  return true;
141
58.8k
}
142
143
}  // namespace jxl