Project
Loading...
Searching...
No Matches
ASoA.h
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
12#ifndef O2_FRAMEWORK_ASOA_H_
13#define O2_FRAMEWORK_ASOA_H_
14
15#include "Framework/Pack.h"
17#include "Headers/DataHeader.h"
20#include "Framework/Traits.h"
26#include <arrow/table.h>
27#include <arrow/array.h>
28#include <arrow/util/config.h>
29#include <gandiva/selection_vector.h>
30#include <array>
31#include <cassert>
32#include <fmt/format.h>
33#include <concepts>
34#include <cstring>
35#include <gsl/span>
36#include <limits>
37
38namespace o2::framework
39{
40using ListVector = std::vector<std::vector<int64_t>>;
41
42std::string cutString(std::string&& str);
43std::string strToUpper(std::string&& str);
44} // namespace o2::framework
45
46namespace o2::soa
47{
48void accessingInvalidIndexFor(const char* getter);
49void dereferenceWithWrongType(const char* getter, const char* target);
50void missingFilterDeclaration(int hash, int ai);
51void notBoundTable(const char* tableName);
52} // namespace o2::soa
53
54namespace o2::soa
55{
57struct TableRef {
58 consteval TableRef()
59 : label_hash{0},
60 desc_hash{0},
61 origin_hash{0},
62 version{0}
63 {
64 }
65 consteval TableRef(uint32_t _label, uint32_t _desc, uint32_t _origin, uint32_t _version)
66 : label_hash{_label},
67 desc_hash{_desc},
68 origin_hash{_origin},
69 version{_version}
70 {
71 }
72 uint32_t label_hash;
73 uint32_t desc_hash;
74 uint32_t origin_hash;
75 uint32_t version;
76
77 constexpr bool operator==(TableRef const& other) const noexcept
78 {
79 return (this->label_hash == other.label_hash) &&
80 (this->desc_hash == other.desc_hash) &&
81 (this->origin_hash == other.origin_hash) &&
82 (this->version == other.version);
83 }
84
85 constexpr bool descriptionCompatible(TableRef const& other) const noexcept
86 {
87 return this->desc_hash == other.desc_hash;
88 }
89
90 constexpr bool descriptionCompatible(uint32_t _desc_hash) const noexcept
91 {
92 return this->desc_hash == _desc_hash;
93 }
94
95 constexpr TableRef(TableRef const&) = default;
96 constexpr TableRef& operator=(TableRef const&) = default;
97 constexpr TableRef(TableRef&&) = default;
98 constexpr TableRef& operator=(TableRef&&) = default;
99};
100
102template <size_t N1, size_t N2, std::array<TableRef, N1> ar1, std::array<TableRef, N2> ar2>
103consteval auto merge()
104{
105 constexpr const int duplicates = std::ranges::count_if(ar2.begin(), ar2.end(), [&](TableRef const& a) { return std::any_of(ar1.begin(), ar1.end(), [&](TableRef const& e) { return e == a; }); });
106 std::array<TableRef, N1 + N2 - duplicates> out;
107
108 auto pos = std::copy(ar1.begin(), ar1.end(), out.begin());
109 std::copy_if(ar2.begin(), ar2.end(), pos, [&](TableRef const& a) { return std::none_of(ar1.begin(), ar1.end(), [&](TableRef const& e) { return e == a; }); });
110 return out;
111}
112
113template <size_t N1, size_t N2, std::array<TableRef, N1> ar1, std::array<TableRef, N2> ar2, typename L>
114consteval auto merge_if(L l)
115{
116 constexpr const int to_remove = std::ranges::count_if(ar1.begin(), ar1.end(), [&](TableRef const& a) { return !l(a); });
117 constexpr const int duplicates = std::ranges::count_if(ar2.begin(), ar2.end(), [&](TableRef const& a) { return std::any_of(ar1.begin(), ar1.end(), [&](TableRef const& e) { return e == a; }) || !l(a); });
118 std::array<TableRef, N1 + N2 - duplicates - to_remove> out;
119
120 auto pos = std::copy_if(ar1.begin(), ar1.end(), out.begin(), [&](TableRef const& a) { return l(a); });
121 std::copy_if(ar2.begin(), ar2.end(), pos, [&](TableRef const& a) { return std::none_of(ar1.begin(), ar1.end(), [&](TableRef const& e) { return e == a; }) && l(a); });
122 return out;
123}
124
125template <size_t N, std::array<TableRef, N> ar, typename L>
126consteval auto remove_if(L l)
127{
128 constexpr const int to_remove = std::ranges::count_if(ar.begin(), ar.end(), [&l](TableRef const& e) { return l(e); });
129 std::array<TableRef, N - to_remove> out;
130 std::copy_if(ar.begin(), ar.end(), out.begin(), [&l](TableRef const& e) { return !l(e); });
131 return out;
132}
133
134template <size_t N1, size_t N2, std::array<TableRef, N1> ar1, std::array<TableRef, N2> ar2>
135consteval auto intersect()
136{
137 constexpr const int duplicates = std::ranges::count_if(ar2.begin(), ar2.end(), [&](TableRef const& a) { return std::any_of(ar1.begin(), ar1.end(), [&](TableRef const& e) { return e == a; }); });
138 std::array<TableRef, duplicates> out;
139 std::copy_if(ar1.begin(), ar1.end(), out.begin(), [](TableRef const& a) { return std::find(ar2.begin(), ar2.end(), a) != ar2.end(); });
140 return out;
141}
142
143template <typename T, typename... Ts>
144consteval auto mergeOriginals()
145 requires(sizeof...(Ts) == 1)
146{
148 return merge<T::originals.size(), T1::originals.size(), T::originals, T1::originals>();
149}
150
151template <typename T, typename... Ts>
152consteval auto mergeOriginals()
153 requires(sizeof...(Ts) > 1)
154{
155 constexpr auto tail = mergeOriginals<Ts...>();
156 return merge<T::originals.size(), tail.size(), T::originals, tail>();
157}
158
159template <typename T, typename... Ts>
160 requires(sizeof...(Ts) == 1)
161consteval auto intersectOriginals()
162{
164 return intersect<T::originals.size(), T1::originals.size(), T::originals, T1::originals>();
165}
166
167template <typename T, typename... Ts>
168 requires(sizeof...(Ts) > 1)
169consteval auto intersectOriginals()
170{
171 constexpr auto tail = intersectOriginals<Ts...>();
172 return intersect<T::originals.size(), tail.size(), T::originals, tail>();
173}
174} // namespace o2::soa
175
176namespace o2::soa
177{
178struct Binding;
179
180template <typename T>
181concept not_void = requires { !std::same_as<T, void>; };
182
184template <typename C>
185concept is_persistent_column = requires(C c) { c.mColumnIterator; };
186
187template <typename C>
189
190template <typename C>
191using is_persistent_column_t = std::conditional_t<is_persistent_column<C>, std::true_type, std::false_type>;
192
193template <typename C>
194concept is_self_index_column = not_void<typename C::self_index_t> && std::same_as<typename C::self_index_t, std::true_type>;
195
196template <typename C>
198 { c.setCurrentRaw(b) } -> std::same_as<bool>;
199 requires std::same_as<decltype(c.mBinding), o2::soa::Binding>;
200};
201
202template <typename C>
203using is_external_index_t = typename std::conditional_t<is_index_column<C>, std::true_type, std::false_type>;
204
205template <typename C>
206using is_self_index_t = typename std::conditional_t<is_self_index_column<C>, std::true_type, std::false_type>;
207} // namespace o2::soa
208
209namespace o2::aod
210{
212template <typename D, typename... Cs>
214 using columns = framework::pack<Cs...>;
218
219 template <typename Key, typename... PCs>
220 static consteval std::array<bool, sizeof...(PCs)> getMap(framework::pack<PCs...>)
221 {
222 return std::array<bool, sizeof...(PCs)>{[]() {
223 if constexpr (requires { PCs::index_targets.size(); }) {
224 return Key::template isIndexTargetOf<PCs::index_targets.size(), PCs::index_targets>();
225 } else {
226 return false;
227 }
228 }()...};
229 }
230
231 template <typename Key>
232 static consteval int getIndexPosToKey()
233 {
235 }
236
237 template <typename Key, size_t N, std::array<bool, N> map>
238 static consteval int getIndexPosToKey_impl()
239 {
240 constexpr const auto pos = std::find(map.begin(), map.end(), true);
241 if constexpr (pos != map.end()) {
242 return std::distance(map.begin(), pos);
243 } else {
244 return -1;
245 }
246 }
247};
248
249template <typename D>
251 using metadata = void;
252};
253
256template <uint32_t H>
257struct Hash {
258 static constexpr uint32_t hash = H;
259 static constexpr char const* const str{""};
260};
261
263template <size_t N, std::array<soa::TableRef, N> ar, typename Key>
264consteval auto filterForKey()
265{
266 constexpr std::array<bool, N> test = []<size_t... Is>(std::index_sequence<Is...>) {
267 return std::array<bool, N>{(Key::template hasOriginal<ar[Is]>() || (o2::aod::MetadataTrait<o2::aod::Hash<ar[Is].desc_hash>>::metadata::template getIndexPosToKey<Key>() >= 0))...};
268 }(std::make_index_sequence<N>());
269 constexpr int correct = std::ranges::count(test.begin(), test.end(), true);
270 std::array<soa::TableRef, correct> out;
271 std::ranges::copy_if(ar.begin(), ar.end(), out.begin(), [&test](soa::TableRef const& r) { return test[std::distance(ar.begin(), std::find(ar.begin(), ar.end(), r))]; });
272 return out;
273}
274
276#define O2HASH(_Str_) \
277 template <> \
278 struct Hash<_Str_ ""_h> { \
279 static constexpr uint32_t hash = _Str_ ""_h; \
280 static constexpr char const* const str{_Str_}; \
281 };
282
284#define O2ORIGIN(_Str_) \
285 template <> \
286 struct Hash<_Str_ ""_h> { \
287 static constexpr header::DataOrigin origin{_Str_}; \
288 static constexpr uint32_t hash = _Str_ ""_h; \
289 static constexpr char const* const str{_Str_}; \
290 };
291
293static inline constexpr uint32_t version(const char* const str)
294{
295 if (str[0] == '\0') {
296 return 0;
297 }
298 size_t len = 0;
299 uint32_t res = 0;
300 while (str[len] != '/' && str[len] != '\0') {
301 ++len;
302 }
303 if (str[len - 1] == '\0') {
304 return -1;
305 }
306 for (auto i = len + 1; str[i] != '\0'; ++i) {
307 res = res * 10 + (int)(str[i] - '0');
308 }
309 return res;
310}
311
313static inline constexpr std::string_view description_str(const char* const str)
314{
315 size_t len = 0;
316 while (len < 15 && str[len] != '/') {
317 ++len;
318 }
319 return std::string_view{str, len};
320}
321
322static inline constexpr header::DataDescription description(const char* const str)
323{
324 size_t len = 0;
325 while (len < 15 && str[len] != '/') {
326 ++len;
327 }
328 char out[16];
329 for (auto i = 0; i < 16; ++i) {
330 out[i] = 0;
331 }
332 std::memcpy(out, str, len);
333 return {out};
334}
335
336// Helpers to get strings from TableRef
337template <soa::TableRef R>
338consteval const char* label()
339{
340 return o2::aod::Hash<R.label_hash>::str;
341}
342
343template <soa::TableRef R>
344consteval const char* origin_str()
345{
346 return o2::aod::Hash<R.origin_hash>::str;
347}
348
349template <soa::TableRef R>
351{
352 return o2::aod::Hash<R.origin_hash>::origin;
353}
354
355template <soa::TableRef R>
356consteval const char* signature()
357{
358 return o2::aod::Hash<R.desc_hash>::str;
359}
360
362template <typename T>
363concept is_aod_hash = requires(T t) { t.hash; t.str; };
364
365template <typename T>
366concept is_origin_hash = is_aod_hash<T> && requires(T t) { t.origin; };
367
369template <soa::TableRef R>
370static constexpr auto sourceSpec()
371{
372 return fmt::format("{}/{}/{}/{}", label<R>(), origin_str<R>(), description_str(signature<R>()), R.version);
373}
374} // namespace o2::aod
375
376namespace o2::soa
377{
378template <aod::is_aod_hash L, aod::is_aod_hash D, aod::is_origin_hash O, typename... Ts>
379class Table;
380
382struct Binding {
383 void const* ptr = nullptr;
384 size_t hash = 0;
385 std::span<TableRef const> refs;
386
387 template <typename T>
388 void bind(T const* table)
389 {
390 ptr = table;
391 hash = o2::framework::TypeIdHelpers::uniqueId<T>();
392 refs = std::span{T::originals};
393 }
394
395 template <typename T>
396 T const* get() const
397 {
398 if (hash == o2::framework::TypeIdHelpers::uniqueId<T>()) {
399 return static_cast<T const*>(ptr);
400 }
401 return nullptr;
402 }
403};
404
405template <typename... C>
407{
408 return std::vector<std::shared_ptr<arrow::Field>>{C::asArrowField()...};
409}
410
411using SelectionVector = std::vector<int64_t>;
412
413template <typename T>
415
416template <typename T>
418
419template <typename T>
421
422template <typename T>
424
425template <typename T>
427
428template <typename T>
429concept is_spawnable_column = std::same_as<typename T::spawnable_t, std::true_type>;
430
431template <typename B, typename E>
433 constexpr static bool value = false;
434};
435
436template <aod::is_aod_hash A, aod::is_aod_hash B>
438 constexpr static bool value = false;
439};
440
441template <typename B, typename E>
443
444template <aod::is_aod_hash A, aod::is_aod_hash B>
446
450struct Chunked {
451 constexpr static bool chunked = true;
452};
453
456struct Flat {
457 constexpr static bool chunked = false;
458};
459
461template <typename T>
462struct unwrap {
463 using type = T;
464};
465
466template <typename T>
467struct unwrap<std::vector<T>> {
468 using type = T;
469};
470
471template <>
472struct unwrap<bool> {
473 using type = char;
474};
475
476template <typename T>
477using unwrap_t = typename unwrap<T>::type;
478
483template <typename T, typename ChunkingPolicy = Chunked>
484class ColumnIterator : ChunkingPolicy
485{
486 static constexpr char SCALE_FACTOR = std::same_as<std::decay_t<T>, bool> ? 3 : 0;
487
488 public:
493 ColumnIterator(arrow::ChunkedArray const* column)
494 : mColumn{column},
495 mCurrent{nullptr},
496 mCurrentPos{nullptr},
497 mLast{nullptr},
498 mFirstIndex{0},
499 mCurrentChunk{0},
500 mOffset{0}
501 {
502 auto array = getCurrentArray();
503 mCurrent = reinterpret_cast<unwrap_t<T> const*>(array->values()->data()) + (mOffset >> SCALE_FACTOR);
504 mLast = mCurrent + array->length();
505 }
506
507 ColumnIterator() = default;
510
513
515 void nextChunk() const
516 {
517 auto previousArray = getCurrentArray();
518 mFirstIndex += previousArray->length();
519
521 auto array = getCurrentArray();
522 mCurrent = reinterpret_cast<unwrap_t<T> const*>(array->values()->data()) + (mOffset >> SCALE_FACTOR) - (mFirstIndex >> SCALE_FACTOR);
523 mLast = mCurrent + array->length() + (mFirstIndex >> SCALE_FACTOR);
524 }
525
526 void prevChunk() const
527 {
528 auto previousArray = getCurrentArray();
529 mFirstIndex -= previousArray->length();
530
532 auto array = getCurrentArray();
533 mCurrent = reinterpret_cast<unwrap_t<T> const*>(array->values()->data()) + (mOffset >> SCALE_FACTOR) - (mFirstIndex >> SCALE_FACTOR);
534 mLast = mCurrent + array->length() + (mFirstIndex >> SCALE_FACTOR);
535 }
536
537 void moveToChunk(int chunk)
538 {
539 if (mCurrentChunk < chunk) {
540 while (mCurrentChunk != chunk) {
541 nextChunk();
542 }
543 } else {
544 while (mCurrentChunk != chunk) {
545 prevChunk();
546 }
547 }
548 }
549
552 {
553 mCurrentChunk = mColumn->num_chunks() - 1;
554 auto array = getCurrentArray();
555 mFirstIndex = mColumn->length() - array->length();
556 mCurrent = reinterpret_cast<unwrap_t<T> const*>(array->values()->data()) + (mOffset >> SCALE_FACTOR) - (mFirstIndex >> SCALE_FACTOR);
557 mLast = mCurrent + array->length() + (mFirstIndex >> SCALE_FACTOR);
558 }
559
560 decltype(auto) operator*() const
561 requires std::same_as<bool, std::decay_t<T>>
562 {
563 checkSkipChunk();
564 return (*(mCurrent - (mOffset >> SCALE_FACTOR) + ((*mCurrentPos + mOffset) >> SCALE_FACTOR)) & (1 << ((*mCurrentPos + mOffset) & 0x7))) != 0;
565 }
566
567 decltype(auto) operator*() const
568 requires((!std::same_as<bool, std::decay_t<T>>) && std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
569 {
570 checkSkipChunk();
571 auto list = std::static_pointer_cast<arrow::ListArray>(mColumn->chunk(mCurrentChunk));
572 auto offset = list->value_offset(*mCurrentPos - mFirstIndex);
573 auto length = list->value_length(*mCurrentPos - mFirstIndex);
574 return gsl::span{mCurrent + mFirstIndex + offset, mCurrent + mFirstIndex + (offset + length)};
575 }
576
577 decltype(auto) operator*() const
578 requires((!std::same_as<bool, std::decay_t<T>>) && !std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
579 {
580 checkSkipChunk();
581 return *(mCurrent + (*mCurrentPos >> SCALE_FACTOR));
582 }
583
584 // Move to the chunk which containts element pos
586 {
587 checkSkipChunk();
588 return *this;
589 }
590
591 mutable unwrap_t<T> const* mCurrent;
592 int64_t const* mCurrentPos;
593 mutable unwrap_t<T> const* mLast;
594 arrow::ChunkedArray const* mColumn;
595 mutable int mFirstIndex;
596 mutable int mCurrentChunk;
597 mutable int mOffset;
598
599 private:
600 void checkSkipChunk() const
601 requires((ChunkingPolicy::chunked == true) && std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
602 {
603 auto list = std::static_pointer_cast<arrow::ListArray>(mColumn->chunk(mCurrentChunk));
604 if (O2_BUILTIN_UNLIKELY(*mCurrentPos - mFirstIndex >= list->length())) {
605 nextChunk();
606 }
607 }
608
609 void checkSkipChunk() const
610 requires((ChunkingPolicy::chunked == true) && !std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
611 {
612 if (O2_BUILTIN_UNLIKELY(((mCurrent + (*mCurrentPos >> SCALE_FACTOR)) >= mLast))) {
613 nextChunk();
614 }
615 }
616
617 void checkSkipChunk() const
618 requires(ChunkingPolicy::chunked == false)
619 {
620 }
622 auto getCurrentArray() const
623 requires(std::same_as<arrow_array_for_t<T>, arrow::FixedSizeListArray>)
624 {
625 std::shared_ptr<arrow::Array> chunkToUse = mColumn->chunk(mCurrentChunk);
626 mOffset = chunkToUse->offset();
627 chunkToUse = std::dynamic_pointer_cast<arrow::FixedSizeListArray>(chunkToUse)->values();
628 return std::static_pointer_cast<arrow_array_for_t<value_for_t<T>>>(chunkToUse);
629 }
630
631 auto getCurrentArray() const
632 requires(std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
633 {
634 std::shared_ptr<arrow::Array> chunkToUse = mColumn->chunk(mCurrentChunk);
635 mOffset = chunkToUse->offset();
636 chunkToUse = std::dynamic_pointer_cast<arrow::ListArray>(chunkToUse)->values();
637 mOffset = chunkToUse->offset();
638 return std::static_pointer_cast<arrow_array_for_t<value_for_t<T>>>(chunkToUse);
639 }
640
641 auto getCurrentArray() const
642 requires(!std::same_as<arrow_array_for_t<T>, arrow::FixedSizeListArray> && !std::same_as<arrow_array_for_t<T>, arrow::ListArray>)
643 {
644 std::shared_ptr<arrow::Array> chunkToUse = mColumn->chunk(mCurrentChunk);
645 mOffset = chunkToUse->offset();
646 return std::static_pointer_cast<arrow_array_for_t<T>>(chunkToUse);
647 }
648};
649
650template <typename T, typename INHERIT>
651struct Column {
652 using inherited_t = INHERIT;
654 : mColumnIterator{it}
655 {
656 }
657
658 Column() = default;
659 Column(Column const&) = default;
660 Column& operator=(Column const&) = default;
661
662 Column(Column&&) = default;
663 Column& operator=(Column&&) = default;
664
665 using type = T;
666 static constexpr const char* const& columnLabel() { return INHERIT::mLabel; }
668 {
669 return mColumnIterator;
670 }
671
672 static auto asArrowField()
673 {
674 return std::make_shared<arrow::Field>(inherited_t::mLabel, framework::expressions::concreteArrowType(framework::expressions::selectArrowType<type>()));
675 }
676
680};
681
684template <typename F, typename INHERIT>
686 using inherited_t = INHERIT;
687
688 static constexpr const char* const& columnLabel() { return INHERIT::mLabel; }
689};
690
691template <typename INHERIT>
693 using inherited_t = INHERIT;
694
695 static constexpr const char* const& columnLabel() { return INHERIT::mLabel; }
696};
697
698template <typename INHERIT>
700 using inherited_t = INHERIT;
701
702 static constexpr const char* const& columnLabel() { return INHERIT::mLabel; }
703};
704
705template <size_t M = 0>
706struct Marker : o2::soa::MarkerColumn<Marker<M>> {
707 using type = size_t;
709 constexpr inline static auto value = M;
710
711 Marker() = default;
712 Marker(Marker const&) = default;
713 Marker(Marker&&) = default;
714
715 Marker& operator=(Marker const&) = default;
716 Marker& operator=(Marker&&) = default;
717
718 Marker(arrow::ChunkedArray const*) {}
719 constexpr inline auto mark()
720 {
721 return value;
722 }
723
724 static constexpr const char* mLabel = "Marker";
725};
726
727template <int64_t START = 0, int64_t END = -1>
728struct Index : o2::soa::IndexColumn<Index<START, END>> {
730 constexpr inline static int64_t start = START;
731 constexpr inline static int64_t end = END;
732
733 Index() = default;
734 Index(Index const&) = default;
735 Index(Index&&) = default;
736
737 Index& operator=(Index const&) = default;
738 Index& operator=(Index&&) = default;
739
740 Index(arrow::ChunkedArray const*)
741 {
742 }
743
744 constexpr inline int64_t rangeStart()
745 {
746 return START;
747 }
748
749 constexpr inline int64_t rangeEnd()
750 {
751 return END;
752 }
753
754 [[nodiscard]] int64_t index() const
755 {
756 return index<0>();
757 }
758
759 [[nodiscard]] int64_t filteredIndex() const
760 {
761 return index<1>();
762 }
763
764 [[nodiscard]] int64_t globalIndex() const
765 {
766 return index<0>() + offsets<0>();
767 }
768
769 template <int N = 0>
770 [[nodiscard]] int64_t index() const
771 {
772 return *std::get<N>(rowIndices);
773 }
774
775 template <int N = 0>
776 [[nodiscard]] int64_t offsets() const
777 {
778 return *std::get<N>(rowOffsets);
779 }
780
781 void setIndices(std::tuple<int64_t const*, int64_t const*> indices)
782 {
784 }
785
786 void setOffsets(std::tuple<uint64_t const*> offsets)
787 {
789 }
790
791 static constexpr const char* mLabel = "Index";
792 using type = int64_t;
793
794 std::tuple<int64_t const*, int64_t const*> rowIndices;
797 std::tuple<uint64_t const*> rowOffsets;
798};
799
800template <typename C>
801concept is_indexing_column = requires(C& c) {
802 c.rowIndices;
803 c.rowOffsets;
804};
805
806template <typename C>
807concept is_dynamic_column = requires(C& c) {
808 c.boundIterators;
809};
810
811template <typename C>
812concept is_marker_column = requires { &C::mark; };
813
814template <typename T>
815using is_dynamic_t = std::conditional_t<is_dynamic_column<T>, std::true_type, std::false_type>;
816
817template <typename T>
819
820template <typename T>
821using is_indexing_t = std::conditional_t<is_indexing_column<T>, std::true_type, std::false_type>;
822
825 int64_t mRowIndex = 0;
827 uint64_t mOffset = 0;
828};
829
831 int64_t const index;
832};
833
835 // We use -1 in the IndexPolicyBase to indicate that the index is
836 // invalid. What will validate the index is the this->setCursor()
837 // which happens below which will properly setup the first index
838 // by remapping the filtered index 0 to whatever unfiltered index
839 // it belongs to.
840 FilteredIndexPolicy(gsl::span<int64_t const> selection, int64_t rows, uint64_t offset = 0)
841 : IndexPolicyBase{-1, offset},
842 mSelectedRows(selection),
843 mMaxSelection(selection.size()),
844 nRows{rows}
845 {
846 this->setCursor(0);
847 }
848
849 void resetSelection(gsl::span<int64_t const> selection)
850 {
851 mSelectedRows = selection;
852 mMaxSelection = selection.size();
853 this->setCursor(0);
854 }
855
861
862 [[nodiscard]] std::tuple<int64_t const*, int64_t const*>
864 {
865 return std::make_tuple(&mRowIndex, &mSelectionRow);
866 }
867
868 [[nodiscard]] std::tuple<uint64_t const*>
870 {
871 return std::make_tuple(&mOffset);
872 }
873
874 void limitRange(int64_t start, int64_t end)
875 {
876 this->setCursor(start);
877 if (end >= 0) {
878 mMaxSelection = std::min(end, mMaxSelection);
879 }
880 }
881
882 void setCursor(int64_t i)
883 {
884 mSelectionRow = i;
885 updateRow();
886 }
887
888 void moveByIndex(int64_t i)
889 {
890 mSelectionRow += i;
891 updateRow();
892 }
893
894 friend bool operator==(FilteredIndexPolicy const& lh, FilteredIndexPolicy const& rh)
895 {
896 return lh.mSelectionRow == rh.mSelectionRow;
897 }
898
899 bool operator==(RowViewSentinel const& sentinel) const
900 {
901 return O2_BUILTIN_UNLIKELY(mSelectionRow == sentinel.index);
902 }
903
908 {
909 this->mSelectionRow = this->mMaxSelection;
910 this->mRowIndex = -1;
911 }
912
913 [[nodiscard]] auto getSelectionRow() const
914 {
915 return mSelectionRow;
916 }
917
918 [[nodiscard]] auto size() const
919 {
920 return mMaxSelection;
921 }
922
923 [[nodiscard]] auto raw_size() const
924 {
925 return nRows;
926 }
927
928 private:
929 inline void updateRow()
930 {
931 this->mRowIndex = O2_BUILTIN_LIKELY(mSelectionRow < mMaxSelection) ? mSelectedRows[mSelectionRow] : -1;
932 }
933 gsl::span<int64_t const> mSelectedRows;
934 int64_t mSelectionRow = 0;
935 int64_t mMaxSelection = 0;
936 int64_t nRows = 0;
937};
938
946
950 DefaultIndexPolicy(int64_t nRows, uint64_t offset)
952 mMaxRow(nRows)
953 {
954 }
955
961
962 void limitRange(int64_t start, int64_t end)
963 {
964 this->setCursor(start);
965 if (end >= 0) {
966 mMaxRow = std::min(end, mMaxRow);
967 }
968 }
969
970 [[nodiscard]] std::tuple<int64_t const*, int64_t const*>
972 {
973 return std::make_tuple(&mRowIndex, &mRowIndex);
974 }
975
976 [[nodiscard]] std::tuple<uint64_t const*>
978 {
979 return std::make_tuple(&mOffset);
980 }
981
982 void setCursor(int64_t i)
983 {
984 this->mRowIndex = i;
985 }
986 void moveByIndex(int64_t i)
987 {
988 this->mRowIndex += i;
989 }
990
992 {
993 this->setCursor(mMaxRow);
994 }
995
996 friend bool operator==(DefaultIndexPolicy const& lh, DefaultIndexPolicy const& rh)
997 {
998 return lh.mRowIndex == rh.mRowIndex;
999 }
1000
1001 bool operator==(RowViewSentinel const& sentinel) const
1002 {
1003 return O2_BUILTIN_UNLIKELY(this->mRowIndex == sentinel.index);
1004 }
1005
1006 [[nodiscard]] auto size() const
1007 {
1008 return mMaxRow;
1009 }
1010
1011 int64_t mMaxRow = 0;
1012};
1013
1014// template <OriginEnc ORIGIN, typename... C>
1015// class Table;
1016
1017template <aod::is_aod_hash L, aod::is_aod_hash D, aod::is_origin_hash O, typename... T>
1018class Table;
1019
1020template <typename T>
1022
1025template <typename C>
1028 arrow::ChunkedArray* second;
1029};
1030
1031template <typename T, typename B>
1032concept can_bind = requires(T&& t) {
1033 { t.B::mColumnIterator };
1034};
1035
1036template <typename... C>
1038
1039template <typename D, typename O, typename IP, typename... C>
1040struct TableIterator : IP, C... {
1041 public:
1042 using self_t = TableIterator<D, O, IP, C...>;
1043 using policy_t = IP;
1048 using bindings_pack_t = decltype([]<typename... Cs>(framework::pack<Cs...>) -> framework::pack<typename Cs::binding_t...> {}(external_index_columns_t{})); // decltype(extractBindings(external_index_columns_t{}));
1049
1050 TableIterator(arrow::ChunkedArray* columnData[sizeof...(C)], IP&& policy)
1051 : IP{policy},
1052 C(columnData[framework::has_type_at_v<C>(all_columns{})])...
1053 {
1054 bind();
1055 }
1056
1057 TableIterator(arrow::ChunkedArray* columnData[sizeof...(C)], IP&& policy)
1058 requires(has_index<C...>)
1059 : IP{policy},
1060 C(columnData[framework::has_type_at_v<C>(all_columns{})])...
1061 {
1062 bind();
1063 // In case we have an index column might need to constrain the actual
1064 // number of rows in the view to the range provided by the index.
1065 // FIXME: we should really understand what happens to an index when we
1066 // have a RowViewFiltered.
1067 this->limitRange(this->rangeStart(), this->rangeEnd());
1068 }
1069
1070 TableIterator() = default;
1072 : IP{static_cast<IP const&>(other)},
1073 C(static_cast<C const&>(other))...
1074 {
1075 bind();
1076 }
1077
1079 {
1080 IP::operator=(static_cast<IP const&>(other));
1081 (void(static_cast<C&>(*this) = static_cast<C>(other)), ...);
1082 bind();
1083 return *this;
1084 }
1085
1087 requires std::same_as<IP, DefaultIndexPolicy>
1088 : IP{static_cast<IP const&>(other)},
1089 C(static_cast<C const&>(other))...
1090 {
1091 bind();
1092 }
1093
1095 {
1096 this->moveByIndex(1);
1097 return *this;
1098 }
1099
1101 {
1102 self_t copy = *this;
1103 this->operator++();
1104 return copy;
1105 }
1106
1108 {
1109 this->moveByIndex(-1);
1110 return *this;
1111 }
1112
1114 {
1115 self_t copy = *this;
1116 this->operator--();
1117 return copy;
1118 }
1119
1121 TableIterator operator+(int64_t inc) const
1122 {
1123 TableIterator copy = *this;
1124 copy.moveByIndex(inc);
1125 return copy;
1126 }
1127
1128 TableIterator operator-(int64_t dec) const
1129 {
1130 return operator+(-dec);
1131 }
1132
1134 {
1135 return *this;
1136 }
1137
1138 template <typename... CL, typename TA>
1140 {
1141 (CL::setCurrent(current), ...);
1142 }
1143
1144 template <typename CL>
1145 auto getCurrent() const
1146 {
1147 return CL::getCurrentRaw();
1148 }
1149
1150 template <typename... Cs>
1152 {
1153 return std::vector<o2::soa::Binding>{static_cast<Cs const&>(*this).getCurrentRaw()...};
1154 }
1155
1156 auto getIndexBindings() const
1157 {
1159 }
1160
1161 template <typename... TA>
1162 void bindExternalIndices(TA*... current)
1163 {
1165 }
1166
1167 template <typename... Cs>
1168 void doSetCurrentIndexRaw(framework::pack<Cs...> p, std::vector<o2::soa::Binding>&& ptrs)
1169 {
1170 (Cs::setCurrentRaw(ptrs[framework::has_type_at_v<Cs>(p)]), ...);
1171 }
1172
1173 template <typename... Cs, typename I>
1175 {
1177 b.bind(ptr);
1178 (Cs::setCurrentRaw(b), ...);
1179 }
1180
1181 void bindExternalIndicesRaw(std::vector<o2::soa::Binding>&& ptrs)
1182 {
1183 doSetCurrentIndexRaw(external_index_columns_t{}, std::forward<std::vector<o2::soa::Binding>>(ptrs));
1184 }
1185
1186 template <typename I>
1187 void bindInternalIndices(I const* table)
1188 {
1190 }
1191
1192 private:
1194 template <typename... PC>
1195 void doMoveToEnd(framework::pack<PC...>)
1196 {
1197 (PC::mColumnIterator.moveToEnd(), ...);
1198 }
1199
1202 void bind()
1203 {
1204 using namespace o2::soa;
1205 auto f = framework::overloaded{
1206 [this]<soa::is_persistent_column T>(T*) -> void { T::mColumnIterator.mCurrentPos = &this->mRowIndex; },
1207 [this]<soa::is_dynamic_column T>(T*) -> void { bindDynamicColumn<T>(typename T::bindings_t{}); },
1208 [this]<typename T>(T*) -> void {},
1209 };
1210 (f(static_cast<C*>(nullptr)), ...);
1211 if constexpr (has_index<C...>) {
1212 this->setIndices(this->getIndices());
1213 this->setOffsets(this->getOffsets());
1214 }
1215 }
1216
1217 template <typename DC, typename... B>
1218 auto bindDynamicColumn(framework::pack<B...>)
1219 {
1220 DC::boundIterators = std::make_tuple(getDynamicBinding<B>()...);
1221 }
1222
1223 // Sometimes dynamic columns are defined for tables in
1224 // the hope that it will be joined / extended with another one which provides
1225 // the full set of bindings. This is to avoid a compilation
1226 // error if constructor for the table or any other thing involving a missing
1227 // binding is preinstanciated.
1228 template <typename B>
1229 requires(can_bind<self_t, B>)
1230 decltype(auto) getDynamicBinding()
1231 {
1232 static_assert(std::same_as<decltype(&(static_cast<B*>(this)->mColumnIterator)), std::decay_t<decltype(B::mColumnIterator)>*>, "foo");
1233 return &(static_cast<B*>(this)->mColumnIterator);
1234 // return static_cast<std::decay_t<decltype(B::mColumnIterator)>*>(nullptr);
1235 }
1236
1237 template <typename B>
1238 decltype(auto) getDynamicBinding()
1239 {
1240 return static_cast<std::decay_t<decltype(B::mColumnIterator)>*>(nullptr);
1241 }
1242};
1243
1245 static std::shared_ptr<arrow::Table> joinTables(std::vector<std::shared_ptr<arrow::Table>>&& tables);
1246 static std::shared_ptr<arrow::Table> concatTables(std::vector<std::shared_ptr<arrow::Table>>&& tables);
1247};
1248
1250template <typename T>
1252
1253template <typename T>
1254concept with_originals = requires {
1255 T::originals.size();
1256};
1257
1258template <typename T>
1259concept with_sources = requires {
1260 T::sources.size();
1261};
1262
1263template <typename T>
1264concept with_base_table = not_void<typename aod::MetadataTrait<o2::aod::Hash<T::ref.desc_hash>>::metadata::base_table_t>;
1265
1266template <size_t N1, std::array<TableRef, N1> os1, size_t N2, std::array<TableRef, N2> os2>
1267consteval bool is_compatible()
1268{
1269 return []<size_t... Is>(std::index_sequence<Is...>) {
1270 return ([]<size_t... Ks>(std::index_sequence<Ks...>) {
1271 constexpr auto h = os1[Is].desc_hash;
1272 using H = o2::aod::Hash<h>;
1273 return (((h == os2[Ks].desc_hash) || is_ng_index_equivalent_v<H, o2::aod::Hash<os2[Ks].desc_hash>>) || ...);
1274 }(std::make_index_sequence<N2>()) ||
1275 ...);
1276 }(std::make_index_sequence<N1>());
1277}
1278
1279template <with_originals T, with_originals B>
1281{
1282 return is_compatible<T::originals.size(), T::originals, B::originals.size(), B::originals>();
1283}
1284
1285template <typename T, typename B>
1286using is_binding_compatible = std::conditional_t<is_binding_compatible_v<T, typename B::binding_t>(), std::true_type, std::false_type>;
1287
1288template <typename L, typename D, typename O, typename Key, typename H, typename... Ts>
1289struct IndexTable;
1290
1291template <typename T>
1293
1294template <soa::is_table T>
1295static constexpr std::string getLabelForTable()
1296{
1297 return std::string{aod::label<std::decay_t<T>::originals[0]>()};
1298}
1299
1300template <soa::is_table T>
1302static constexpr std::string getLabelFromType()
1303{
1304 return getLabelForTable<T>();
1305}
1306
1307template <soa::is_iterator T>
1308static constexpr std::string getLabelFromType()
1309{
1310 return getLabelForTable<typename std::decay_t<T>::parent_t>();
1311}
1312
1313template <soa::is_index_table T>
1314static constexpr std::string getLabelFromType()
1315{
1316 return getLabelForTable<typename std::decay_t<T>::first_t>();
1317}
1318template <soa::with_base_table T>
1319static constexpr std::string getLabelFromType()
1320{
1321 return getLabelForTable<typename aod::MetadataTrait<o2::aod::Hash<T::ref.desc_hash>>::metadata::base_table_t>();
1322}
1323
1324template <typename... C>
1325static constexpr auto hasColumnForKey(framework::pack<C...>, std::string const& key)
1326{
1327 return ((C::inherited_t::mLabel == key) || ...);
1328}
1329
1330template <TableRef ref>
1331static constexpr std::pair<bool, std::string> hasKey(std::string const& key)
1332{
1333 return {hasColumnForKey(typename aod::MetadataTrait<o2::aod::Hash<ref.desc_hash>>::metadata::columns{}, key), aod::label<ref>()};
1334}
1335
1336template <typename... C>
1337static constexpr auto haveKey(framework::pack<C...>, std::string const& key)
1338{
1339 return std::vector{hasKey<C>(key)...};
1340}
1341
1342void notFoundColumn(const char* label, const char* key);
1343void missingOptionalPreslice(const char* label, const char* key);
1344
1345template <with_originals T, bool OPT = false>
1346static constexpr std::string getLabelFromTypeForKey(std::string const& key)
1347{
1348 if constexpr (T::originals.size() == 1) {
1349 auto locate = hasKey<T::originals[0]>(key);
1350 if (locate.first) {
1351 return locate.second;
1352 }
1353 } else {
1354 auto locate = [&]<size_t... Is>(std::index_sequence<Is...>) {
1355 return std::vector{hasKey<T::originals[Is]>(key)...};
1356 }(std::make_index_sequence<T::originals.size()>{});
1357 auto it = std::find_if(locate.begin(), locate.end(), [](auto const& x) { return x.first; });
1358 if (it != locate.end()) {
1359 return it->second;
1360 }
1361 }
1362 if constexpr (!OPT) {
1363 notFoundColumn(getLabelFromType<std::decay_t<T>>().data(), key.data());
1364 } else {
1365 return "[MISSING]";
1366 }
1368}
1369
1370template <typename B, typename... C>
1371consteval static bool hasIndexTo(framework::pack<C...>&&)
1372{
1373 return (o2::soa::is_binding_compatible_v<B, typename C::binding_t>() || ...);
1374}
1375
1376template <typename B, typename... C>
1377consteval static bool hasSortedIndexTo(framework::pack<C...>&&)
1378{
1379 return ((C::sorted && o2::soa::is_binding_compatible_v<B, typename C::binding_t>()) || ...);
1380}
1381
1382template <typename B, typename Z>
1383consteval static bool relatedByIndex()
1384{
1385 return hasIndexTo<B>(typename Z::table_t::external_index_columns_t{});
1386}
1387
1388template <typename B, typename Z>
1389consteval static bool relatedBySortedIndex()
1390{
1391 return hasSortedIndexTo<B>(typename Z::table_t::external_index_columns_t{});
1392}
1393} // namespace o2::soa
1394
1395namespace o2::framework
1396{
1397
1399 const std::string binding;
1401
1402 bool isMissing() const;
1403 StringPair const& getBindingKey() const;
1404};
1405
1407 void updateSliceInfo(SliceInfoPtr&& si);
1408
1410 std::shared_ptr<arrow::Table> getSliceFor(int value, std::shared_ptr<arrow::Table> const& input, uint64_t& offset) const;
1411};
1412
1415
1417 gsl::span<const int64_t> getSliceFor(int value) const;
1418};
1419
1420template <typename T, typename Policy, bool OPT = false>
1421struct PresliceBase : public Policy {
1422 constexpr static bool optional = OPT;
1423 using target_t = T;
1424 using policy_t = Policy;
1425 const std::string binding;
1426
1428 : Policy{PreslicePolicyBase{{o2::soa::getLabelFromTypeForKey<T, OPT>(std::string{index_.name})}, std::make_pair(o2::soa::getLabelFromTypeForKey<T, OPT>(std::string{index_.name}), std::string{index_.name})}, {}}
1429 {
1430 }
1431
1432 std::shared_ptr<arrow::Table> getSliceFor(int value, std::shared_ptr<arrow::Table> const& input, uint64_t& offset) const
1433 {
1434 if constexpr (OPT) {
1435 if (Policy::isMissing()) {
1436 return nullptr;
1437 }
1438 }
1439 return Policy::getSliceFor(value, input, offset);
1440 }
1441
1442 gsl::span<const int64_t> getSliceFor(int value) const
1443 {
1444 if constexpr (OPT) {
1445 if (Policy::isMissing()) {
1446 return {};
1447 }
1448 }
1449 return Policy::getSliceFor(value);
1450 }
1451};
1452
1453template <typename T>
1455template <typename T>
1457template <typename T>
1459template <typename T>
1461
1462template <typename T>
1463concept is_preslice = std::derived_from<T, PreslicePolicyBase>;
1464
1465} // namespace o2::framework
1466
1467namespace o2::soa
1468{
1469template <soa::is_table T>
1470class FilteredBase;
1471template <typename T>
1472class Filtered;
1473
1474template <typename T>
1475concept has_filtered_policy = not_void<typename T::policy_t> && std::same_as<typename T::policy_t, soa::FilteredIndexPolicy>;
1476
1477template <typename T>
1479
1480template <typename T>
1482
1483// FIXME: compatbility declaration to be removed
1484template <typename T>
1486
1487template <typename T>
1489
1490template <typename T>
1492
1494template <typename... Is>
1495static consteval auto extractBindings(framework::pack<Is...>)
1496{
1497 return framework::pack<typename Is::binding_t...>{};
1498}
1499
1501
1502template <typename T, typename C, typename Policy, bool OPT>
1503 requires std::same_as<Policy, framework::PreslicePolicySorted> && (o2::soa::is_binding_compatible_v<C, T>())
1504auto doSliceBy(T const* table, o2::framework::PresliceBase<C, Policy, OPT> const& container, int value)
1505{
1506 if constexpr (OPT) {
1507 if (container.isMissing()) {
1508 missingOptionalPreslice(getLabelFromType<std::decay_t<T>>().data(), container.bindingKey.second.c_str());
1509 }
1510 }
1511 uint64_t offset = 0;
1512 auto out = container.getSliceFor(value, table->asArrowTable(), offset);
1513 auto t = typename T::self_t({out}, offset);
1514 table->copyIndexBindings(t);
1515 t.bindInternalIndicesTo(table);
1516 return t;
1517}
1518
1519template <soa::is_filtered_table T>
1520auto doSliceByHelper(T const* table, gsl::span<const int64_t> const& selection)
1521{
1522 auto t = soa::Filtered<typename T::base_t>({table->asArrowTable()}, selection);
1523 table->copyIndexBindings(t);
1524 t.bindInternalIndicesTo(table);
1525 t.intersectWithSelection(table->getSelectedRows()); // intersect filters
1526 return t;
1527}
1528
1529template <soa::is_table T>
1530 requires(!soa::is_filtered_table<T>)
1531auto doSliceByHelper(T const* table, gsl::span<const int64_t> const& selection)
1532{
1533 auto t = soa::Filtered<T>({table->asArrowTable()}, selection);
1534 table->copyIndexBindings(t);
1535 t.bindInternalIndicesTo(table);
1536 return t;
1537}
1538
1539template <typename T, typename C, typename Policy, bool OPT>
1540 requires std::same_as<Policy, framework::PreslicePolicyGeneral> && (o2::soa::is_binding_compatible_v<C, T>())
1541auto doSliceBy(T const* table, o2::framework::PresliceBase<C, Policy, OPT> const& container, int value)
1542{
1543 if constexpr (OPT) {
1544 if (container.isMissing()) {
1545 missingOptionalPreslice(getLabelFromType<std::decay_t<T>>().data(), container.bindingKey.second.c_str());
1546 }
1547 }
1548 auto selection = container.getSliceFor(value);
1549 return doSliceByHelper(table, selection);
1550}
1551
1552SelectionVector sliceSelection(gsl::span<int64_t const> const& mSelectedRows, int64_t nrows, uint64_t offset);
1553
1554template <soa::is_filtered_table T>
1555auto prepareFilteredSlice(T const* table, std::shared_ptr<arrow::Table> slice, uint64_t offset)
1556{
1557 if (offset >= static_cast<uint64_t>(table->tableSize())) {
1558 Filtered<typename T::base_t> fresult{{{slice}}, SelectionVector{}, 0};
1559 table->copyIndexBindings(fresult);
1560 return fresult;
1561 }
1562 auto slicedSelection = sliceSelection(table->getSelectedRows(), slice->num_rows(), offset);
1563 Filtered<typename T::base_t> fresult{{{slice}}, std::move(slicedSelection), offset};
1564 table->copyIndexBindings(fresult);
1565 return fresult;
1566}
1567
1568template <soa::is_filtered_table T, typename C, bool OPT>
1569 requires(o2::soa::is_binding_compatible_v<C, T>())
1571{
1572 if constexpr (OPT) {
1573 if (container.isMissing()) {
1574 missingOptionalPreslice(getLabelFromType<T>().data(), container.bindingKey.second.c_str());
1575 }
1576 }
1577 uint64_t offset = 0;
1578 auto slice = container.getSliceFor(value, table->asArrowTable(), offset);
1579 return prepareFilteredSlice(table, slice, offset);
1580}
1581
1582template <typename T>
1584{
1585 auto localCache = cache.ptr->getCacheFor({o2::soa::getLabelFromTypeForKey<T>(node.name), node.name});
1586 auto [offset, count] = localCache.getSliceFor(value);
1587 auto t = typename T::self_t({table->asArrowTable()->Slice(static_cast<uint64_t>(offset), count)}, static_cast<uint64_t>(offset));
1588 table->copyIndexBindings(t);
1589 return t;
1590}
1591
1592template <typename T>
1594{
1595 auto localCache = cache.ptr->getCacheFor({o2::soa::getLabelFromTypeForKey<T>(node.name), node.name});
1596 auto [offset, count] = localCache.getSliceFor(value);
1597 auto slice = table->asArrowTable()->Slice(static_cast<uint64_t>(offset), count);
1598 return prepareFilteredSlice(table, slice, offset);
1599}
1600
1601template <typename T>
1603{
1604 auto localCache = cache.ptr->getCacheUnsortedFor({o2::soa::getLabelFromTypeForKey<T>(node.name), node.name});
1605 if constexpr (soa::is_filtered_table<T>) {
1606 auto t = typename T::self_t({table->asArrowTable()}, localCache.getSliceFor(value));
1607 t.intersectWithSelection(table->getSelectedRows());
1608 table->copyIndexBindings(t);
1609 return t;
1610 } else {
1611 auto t = Filtered<T>({table->asArrowTable()}, localCache.getSliceFor(value));
1612 table->copyIndexBindings(t);
1613 return t;
1614 }
1615}
1616
1617template <with_originals T>
1619{
1620 return Filtered<T>({t.asArrowTable()}, selectionToVector(framework::expressions::createSelection(t.asArrowTable(), f)));
1621}
1622
1623arrow::ChunkedArray* getIndexFromLabel(arrow::Table* table, std::string_view label);
1624
1625template <typename D, typename O, typename IP, typename... C>
1626consteval auto base_iter(framework::pack<C...>&&) -> TableIterator<D, O, IP, C...>
1627{
1628}
1629
1630template <TableRef ref, typename... Ts>
1631 requires((sizeof...(Ts) > 0) && (soa::is_column<Ts> && ...))
1632consteval auto getColumns()
1633{
1634 return framework::pack<Ts...>{};
1635}
1636
1637template <TableRef ref, typename... Ts>
1638 requires((sizeof...(Ts) > 0) && !(soa::is_column<Ts> || ...) && (ref.origin_hash == "CONC"_h))
1639consteval auto getColumns()
1640{
1641 return framework::full_intersected_pack_t<typename Ts::columns_t...>{};
1642}
1643
1644template <TableRef ref, typename... Ts>
1645 requires((sizeof...(Ts) > 0) && !(soa::is_column<Ts> || ...) && (ref.origin_hash != "CONC"_h))
1646consteval auto getColumns()
1647{
1648 return framework::concatenated_pack_unique_t<typename Ts::columns_t...>{};
1649}
1650
1651template <TableRef ref, typename... Ts>
1652 requires(sizeof...(Ts) == 0 && soa::has_metadata<aod::MetadataTrait<o2::aod::Hash<ref.desc_hash>>>)
1653consteval auto getColumns()
1654{
1655 return typename aod::MetadataTrait<o2::aod::Hash<ref.desc_hash>>::metadata::columns{};
1656}
1657
1658template <TableRef ref, typename... Ts>
1659 requires((sizeof...(Ts) == 0) || (o2::soa::is_column<Ts> && ...))
1660consteval auto computeOriginals()
1661{
1662 return std::array<TableRef, 1>{ref};
1663}
1664
1665template <TableRef ref, typename... Ts>
1666 requires((sizeof...(Ts) > 0) && (!o2::soa::is_column<Ts> || ...))
1667consteval auto computeOriginals()
1668{
1669 return o2::soa::mergeOriginals<Ts...>();
1670}
1671
1674template <aod::is_aod_hash L, aod::is_aod_hash D, aod::is_origin_hash O, typename... Ts>
1676{
1677 public:
1678 static constexpr const auto ref = TableRef{L::hash, D::hash, O::hash, o2::aod::version(D::str)};
1679 using self_t = Table<L, D, O, Ts...>;
1681
1682 static constexpr const auto originals = computeOriginals<ref, Ts...>();
1683
1684 template <size_t N, std::array<TableRef, N> bindings>
1685 requires(ref.origin_hash == "CONC"_h)
1686 static consteval auto isIndexTargetOf()
1687 {
1688 return false;
1689 }
1690
1691 template <size_t N, std::array<TableRef, N> bindings>
1692 requires(ref.origin_hash == "JOIN"_h)
1693 static consteval auto isIndexTargetOf()
1694 {
1695 return std::find_if(self_t::originals.begin(), self_t::originals.end(),
1696 [](TableRef const& r) {
1697 return std::find(bindings.begin(), bindings.end(), r) != bindings.end();
1698 }) != self_t::originals.end();
1699 }
1700
1701 template <size_t N, std::array<TableRef, N> bindings>
1702 requires(!(ref.origin_hash == "CONC"_h || ref.origin_hash == "JOIN"_h))
1703 static consteval auto isIndexTargetOf()
1704 {
1705 return std::find(bindings.begin(), bindings.end(), self_t::ref) != bindings.end();
1706 }
1707
1708 template <TableRef r>
1709 static consteval bool hasOriginal()
1710 {
1711 return std::find_if(originals.begin(), originals.end(), [](TableRef const& o) { return o.desc_hash == r.desc_hash; }) != originals.end();
1712 }
1713
1714 using columns_t = decltype(getColumns<ref, Ts...>());
1715
1718
1721 template <typename IP>
1722 using base_iterator = decltype(base_iter<D, O, IP>(columns_t{}));
1723
1724 template <typename IP, typename Parent, typename... T>
1726 using columns_t = typename Parent::columns_t;
1727 using external_index_columns_t = typename Parent::external_index_columns_t;
1729 // static constexpr const std::array<TableRef, sizeof...(T)> originals{T::ref...};
1730 static constexpr auto originals = Parent::originals;
1731 using policy_t = IP;
1732 using parent_t = Parent;
1733
1735
1736 TableIteratorBase(arrow::ChunkedArray* columnData[framework::pack_size(columns_t{})], IP&& policy)
1737 : base_iterator<IP>(columnData, std::forward<decltype(policy)>(policy))
1738 {
1739 }
1740
1741 template <typename P, typename... Os>
1743 requires(P::ref.desc_hash == Parent::ref.desc_hash)
1744 {
1745 static_cast<base_iterator<IP>&>(*this) = static_cast<base_iterator<IP>>(other);
1746 return *this;
1747 }
1748
1749 template <typename P>
1751 {
1752 static_cast<base_iterator<IP>&>(*this) = static_cast<base_iterator<IP>>(other);
1753 return *this;
1754 }
1755
1756 template <typename P>
1758 requires std::same_as<IP, DefaultIndexPolicy>
1759 {
1760 static_cast<base_iterator<IP>&>(*this) = static_cast<base_iterator<FilteredIndexPolicy>>(other);
1761 return *this;
1762 }
1763
1764 template <typename P, typename O1, typename... Os>
1766 requires(P::ref.desc_hash == Parent::ref.desc_hash)
1767 {
1768 *this = other;
1769 }
1770
1771 template <typename P, typename O1, typename... Os>
1773 requires(P::ref.desc_hash == Parent::ref.desc_hash)
1774 {
1775 *this = other;
1776 }
1777
1778 template <typename P>
1783
1784 template <typename P>
1786 {
1787 *this = other;
1788 }
1789
1790 template <typename P>
1792 requires std::same_as<IP, DefaultIndexPolicy>
1793 {
1794 *this = other;
1795 }
1796
1798 {
1799 this->mRowIndex = other.index;
1800 return *this;
1801 }
1802 template <typename P>
1804 {
1805 this->mRowIndex = other.mRowIndex;
1806 }
1807
1808 template <typename P, typename... Os>
1810 requires std::same_as<typename P::table_t, typename Parent::table_t>
1811 {
1812 this->mRowIndex = other.mRowIndex;
1813 }
1814
1815 template <typename TI>
1816 auto getId() const
1817 {
1818 using decayed = std::decay_t<TI>;
1819 if constexpr (framework::has_type<decayed>(bindings_pack_t{})) { // index to another table
1820 constexpr auto idx = framework::has_type_at_v<decayed>(bindings_pack_t{});
1822 } else if constexpr (std::same_as<decayed, Parent>) { // self index
1823 return this->globalIndex();
1824 } else if constexpr (is_indexing_column<decayed>) { // soa::Index<>
1825 return this->globalIndex();
1826 } else {
1827 return static_cast<int32_t>(-1);
1828 }
1829 }
1830
1831 template <typename CD, typename... CDArgs>
1832 auto getDynamicColumn() const
1833 {
1834 using decayed = std::decay_t<CD>;
1835 static_assert(is_dynamic_t<decayed>(), "Requested column is not a dynamic column");
1836 return static_cast<decayed>(*this).template getDynamicValue<CDArgs...>();
1837 }
1838
1839 template <typename B, typename CC>
1840 auto getValue() const
1841 {
1842 using COL = std::decay_t<CC>;
1843 static_assert(is_dynamic_t<COL>() || soa::is_persistent_column<COL>, "Should be persistent or dynamic column with no argument that has a return type convertable to float");
1844 return static_cast<B>(static_cast<COL>(*this).get());
1845 }
1846
1847 template <typename B, typename... CCs>
1848 std::array<B, sizeof...(CCs)> getValues() const
1849 {
1850 static_assert(std::same_as<B, float> || std::same_as<B, double>, "The common return type should be float or double");
1851 return {getValue<B, CCs>()...};
1852 }
1853
1854 using IP::size;
1855
1856 using base_iterator<IP>::operator++;
1857
1859 TableIteratorBase operator+(int64_t inc) const
1860 {
1861 TableIteratorBase copy = *this;
1862 copy.moveByIndex(inc);
1863 return copy;
1864 }
1865
1866 TableIteratorBase operator-(int64_t dec) const
1867 {
1868 return operator+(-dec);
1869 }
1870
1872 {
1873 return *this;
1874 }
1875 };
1876
1877 template <typename IP, typename Parent, typename... T>
1879
1880 template <typename IP, typename Parent>
1881 static consteval auto full_iter()
1882 {
1883 if constexpr (sizeof...(Ts) == 0) {
1885 } else {
1886 if constexpr ((o2::soa::is_column<Ts> && ...)) {
1888 } else {
1889 return iterator_template<IP, Parent, Ts...>{};
1890 }
1891 }
1892 }
1893
1894 template <typename IP, typename Parent>
1895 using iterator_template_o = decltype(full_iter<IP, Parent>());
1896
1899
1903
1904 static constexpr auto hashes()
1905 {
1906 return []<typename... C>(framework::pack<C...>) { return std::set{{o2::framework::TypeIdHelpers::uniqueId<C>()...}}; }(columns_t{});
1907 }
1908
1909 Table(std::shared_ptr<arrow::Table> table, uint64_t offset = 0)
1910 : mTable(table),
1911 mEnd{table->num_rows()},
1912 mOffset(offset)
1913 {
1914 if (mTable->num_rows() == 0) {
1915 for (size_t ci = 0; ci < framework::pack_size(columns_t{}); ++ci) {
1916 mColumnChunks[ci] = nullptr;
1917 }
1918 mBegin = mEnd;
1919 } else {
1920 auto lookups = [this]<typename... C>(framework::pack<C...>) -> std::array<arrow::ChunkedArray*, framework::pack_size(columns_t{})> { return {lookupColumn<C>()...}; }(columns_t{});
1921 for (size_t ci = 0; ci < framework::pack_size(columns_t{}); ++ci) {
1922 mColumnChunks[ci] = lookups[ci];
1923 }
1924 mBegin = unfiltered_iterator{mColumnChunks, {table->num_rows(), offset}};
1925 mBegin.bindInternalIndices(this);
1926 }
1927 }
1928
1929 Table(std::vector<std::shared_ptr<arrow::Table>>&& tables, uint64_t offset = 0)
1930 requires(ref.origin_hash != "CONC"_h)
1931 : Table(ArrowHelpers::joinTables(std::move(tables)), offset)
1932 {
1933 }
1934
1935 Table(std::vector<std::shared_ptr<arrow::Table>>&& tables, uint64_t offset = 0)
1936 requires(ref.origin_hash == "CONC"_h)
1937 : Table(ArrowHelpers::concatTables(std::move(tables)), offset)
1938 {
1939 }
1940
1941 template <typename Key>
1942 inline arrow::ChunkedArray* getIndexToKey()
1943 {
1944 constexpr auto map = []<typename... Cs>(framework::pack<Cs...>) {
1945 return std::array<bool, sizeof...(Cs)>{[]() {
1946 if constexpr (requires { Cs::index_targets.size(); }) {
1947 return Key::template isIndexTargetOf<Cs::index_targets.size(), Cs::index_targets>();
1948 } else {
1949 return false;
1950 }
1951 }()...};
1953 constexpr auto pos = std::find(map.begin(), map.end(), true);
1954 if constexpr (pos != map.end()) {
1955 return mColumnChunks[std::distance(map.begin(), pos)];
1956 } else {
1957 static_assert(framework::always_static_assert_v<Key>, "This table does not have an index to given Key");
1958 }
1959 }
1960
1962 {
1963 return mBegin;
1964 }
1965
1966 auto const& cached_begin() const
1967 {
1968 return mBegin;
1969 }
1970
1972 {
1973 return unfiltered_iterator(mBegin);
1974 }
1975
1977 {
1978 return RowViewSentinel{mEnd};
1979 }
1980
1981 filtered_iterator filtered_begin(gsl::span<int64_t const> selection)
1982 {
1983 // Note that the FilteredIndexPolicy will never outlive the selection which
1984 // is held by the table, so we are safe passing the bare pointer. If it does it
1985 // means that the iterator on a table is outliving the table itself, which is
1986 // a bad idea.
1987 return filtered_iterator(mColumnChunks, {selection, mTable->num_rows(), mOffset});
1988 }
1989
1990 iterator iteratorAt(uint64_t i) const
1991 {
1992 return rawIteratorAt(i);
1993 }
1994
1996 {
1997 auto it = mBegin;
1998 it.setCursor(i);
1999 return it;
2000 }
2001
2003 {
2004 return unfiltered_const_iterator(mBegin);
2005 }
2006
2007 [[nodiscard]] RowViewSentinel end() const
2008 {
2009 return RowViewSentinel{mEnd};
2010 }
2011
2013 [[nodiscard]] std::shared_ptr<arrow::Table> asArrowTable() const
2014 {
2015 return mTable;
2016 }
2018 auto offset() const
2019 {
2020 return mOffset;
2021 }
2023 [[nodiscard]] int64_t size() const
2024 {
2025 return mTable->num_rows();
2026 }
2027
2028 [[nodiscard]] int64_t tableSize() const
2029 {
2030 return size();
2031 }
2032
2035 template <typename... TA>
2036 void bindExternalIndices(TA*... current)
2037 {
2038 mBegin.bindExternalIndices(current...);
2039 }
2040
2041 template <typename I>
2043 {
2044 mBegin.bindInternalIndices(ptr);
2045 }
2046
2051
2052 template <typename... Cs>
2054 {
2055 (static_cast<Cs>(mBegin).setCurrentRaw(binding), ...);
2056 }
2057
2058 void bindExternalIndicesRaw(std::vector<o2::soa::Binding>&& ptrs)
2059 {
2060 mBegin.bindExternalIndicesRaw(std::forward<std::vector<o2::soa::Binding>>(ptrs));
2061 }
2062
2063 template <typename T, typename... Cs>
2065 {
2066 dest.bindExternalIndicesRaw(mBegin.getIndexBindings());
2067 }
2068
2069 template <typename T>
2070 void copyIndexBindings(T& dest) const
2071 {
2073 }
2074
2076 {
2077 auto t = o2::soa::select(*this, f);
2079 return t;
2080 }
2081
2083 {
2084 return doSliceByCached(this, node, value, cache);
2085 }
2086
2088 {
2089 return doSliceByCachedUnsorted(this, node, value, cache);
2090 }
2091
2092 template <typename T1, typename Policy, bool OPT>
2094 {
2095 return doSliceBy(this, container, value);
2096 }
2097
2098 auto rawSlice(uint64_t start, uint64_t end) const
2099 {
2100 return self_t{mTable->Slice(start, end - start + 1), start};
2101 }
2102
2103 auto emptySlice() const
2104 {
2105 return self_t{mTable->Slice(0, 0), 0};
2106 }
2107
2108 private:
2109 template <typename T>
2110 arrow::ChunkedArray* lookupColumn()
2111 {
2112 if constexpr (soa::is_persistent_column<T>) {
2113 auto label = T::columnLabel();
2114 return getIndexFromLabel(mTable.get(), label);
2115 } else {
2116 return nullptr;
2117 }
2118 }
2119 std::shared_ptr<arrow::Table> mTable = nullptr;
2120 uint64_t mOffset = 0;
2121 // Cached pointers to the ChunkedArray associated to a column
2122 arrow::ChunkedArray* mColumnChunks[framework::pack_size(columns_t{})];
2123 RowViewSentinel mEnd;
2124 iterator mBegin;
2125};
2126
2127template <uint32_t D, soa::is_column... C>
2129
2130void getterNotFound(const char* targetColumnLabel);
2131void emptyColumnLabel();
2132
2133namespace row_helpers
2134{
2135template <soa::is_persistent_column... Cs>
2136std::array<arrow::ChunkedArray*, sizeof...(Cs)> getArrowColumns(arrow::Table* table, framework::pack<Cs...>)
2137{
2138 return std::array<arrow::ChunkedArray*, sizeof...(Cs)>{o2::soa::getIndexFromLabel(table, Cs::columnLabel())...};
2139}
2140
2141template <soa::is_persistent_column... Cs>
2142std::array<std::shared_ptr<arrow::Array>, sizeof...(Cs)> getChunks(arrow::Table* table, framework::pack<Cs...>, uint64_t ci)
2143{
2144 return std::array<std::shared_ptr<arrow::Array>, sizeof...(Cs)>{o2::soa::getIndexFromLabel(table, Cs::columnLabel())->chunk(ci)...};
2145}
2146
2147template <typename T, soa::is_persistent_column C>
2148typename C::type getSingleRowData(arrow::Table* table, T& rowIterator, uint64_t ci = std::numeric_limits<uint64_t>::max(), uint64_t ai = std::numeric_limits<uint64_t>::max(), uint64_t globalIndex = std::numeric_limits<uint64_t>::max())
2149{
2150 if (ci == std::numeric_limits<uint64_t>::max() || ai == std::numeric_limits<uint64_t>::max()) {
2151 auto colIterator = static_cast<C>(rowIterator).getIterator();
2152 ci = colIterator.mCurrentChunk;
2153 ai = *(colIterator.mCurrentPos) - colIterator.mFirstIndex;
2154 }
2155 return std::static_pointer_cast<o2::soa::arrow_array_for_t<typename C::type>>(o2::soa::getIndexFromLabel(table, C::columnLabel())->chunk(ci))->raw_values()[ai];
2156}
2157
2158template <typename T, soa::is_dynamic_column C>
2159typename C::type getSingleRowData(arrow::Table*, T& rowIterator, uint64_t ci = std::numeric_limits<uint64_t>::max(), uint64_t ai = std::numeric_limits<uint64_t>::max(), uint64_t globalIndex = std::numeric_limits<uint64_t>::max())
2160{
2161 if (globalIndex != std::numeric_limits<uint64_t>::max() && globalIndex != *std::get<0>(rowIterator.getIndices())) {
2162 rowIterator.setCursor(globalIndex);
2163 }
2164 return rowIterator.template getDynamicColumn<C>();
2165}
2166
2167template <typename T, soa::is_index_column C>
2168typename C::type getSingleRowData(arrow::Table*, T& rowIterator, uint64_t ci = std::numeric_limits<uint64_t>::max(), uint64_t ai = std::numeric_limits<uint64_t>::max(), uint64_t globalIndex = std::numeric_limits<uint64_t>::max())
2169{
2170 if (globalIndex != std::numeric_limits<uint64_t>::max() && globalIndex != *std::get<0>(rowIterator.getIndices())) {
2171 rowIterator.setCursor(globalIndex);
2172 }
2173 return rowIterator.template getId<C>();
2174}
2175
2176template <typename T, typename... Cs>
2177std::tuple<typename Cs::type...> getRowData(arrow::Table* table, T rowIterator, uint64_t ci = std::numeric_limits<uint64_t>::max(), uint64_t ai = std::numeric_limits<uint64_t>::max(), uint64_t globalIndex = std::numeric_limits<uint64_t>::max())
2178{
2179 return std::make_tuple(getSingleRowData<T, Cs>(table, rowIterator, ci, ai, globalIndex)...);
2180}
2181
2182namespace
2183{
2184template <typename R, typename T, typename C>
2185R getColumnValue(const T& rowIterator)
2186{
2187 return static_cast<R>(static_cast<C>(rowIterator).get());
2188}
2189
2190template <typename R, typename T>
2191using ColumnGetterFunction = R (*)(const T&);
2192
2193template <typename T, typename R>
2195 // lambda is callable without additional free args
2196 framework::pack_size(typename T::bindings_t{}) == framework::pack_size(typename T::callable_t::args{}) &&
2197 requires(T t) {
2198 { t.get() } -> std::convertible_to<R>;
2199 };
2200
2201template <typename T, typename R>
2202concept persistent_with_common_getter = is_persistent_v<T> && requires(T t) {
2203 { t.get() } -> std::convertible_to<R>;
2204};
2205
2206template <typename R, typename T, persistent_with_common_getter<R> C>
2207ColumnGetterFunction<R, T> createGetterPtr(const std::string_view& targetColumnLabel)
2208{
2209 return targetColumnLabel == C::columnLabel() ? &getColumnValue<R, T, C> : nullptr;
2210}
2211
2212template <typename R, typename T, dynamic_with_common_getter<R> C>
2213ColumnGetterFunction<R, T> createGetterPtr(const std::string_view& targetColumnLabel)
2214{
2215 std::string_view columnLabel(C::columnLabel());
2216
2217 // allows user to use consistent formatting (with prefix) of all column labels
2218 // by default there isn't 'f' prefix for dynamic column labels
2219 if (targetColumnLabel.starts_with("f") && targetColumnLabel.substr(1) == columnLabel) {
2220 return &getColumnValue<R, T, C>;
2221 }
2222
2223 // check also exact match if user is aware of prefix missing
2224 if (targetColumnLabel == columnLabel) {
2225 return &getColumnValue<R, T, C>;
2226 }
2227
2228 return nullptr;
2229}
2230
2231template <typename R, typename T, typename... Cs>
2232ColumnGetterFunction<R, T> getColumnGetterByLabel(o2::framework::pack<Cs...>, const std::string_view& targetColumnLabel)
2233{
2234 ColumnGetterFunction<R, T> func;
2235
2236 (void)((func = createGetterPtr<R, T, Cs>(targetColumnLabel), func) || ...);
2237
2238 if (!func) {
2239 getterNotFound(targetColumnLabel.data());
2240 }
2241
2242 return func;
2243}
2244
2245template <typename T, typename R>
2246using with_common_getter_t = typename std::conditional<persistent_with_common_getter<T, R> || dynamic_with_common_getter<T, R>, std::true_type, std::false_type>::type;
2247} // namespace
2248
2249template <typename R, typename T>
2250ColumnGetterFunction<R, typename T::iterator> getColumnGetterByLabel(const std::string_view& targetColumnLabel)
2251{
2252 using TypesWithCommonGetter = o2::framework::selected_pack_multicondition<with_common_getter_t, framework::pack<R>, typename T::columns_t>;
2253
2254 if (targetColumnLabel.size() == 0) {
2256 }
2257
2258 return getColumnGetterByLabel<R, typename T::iterator>(TypesWithCommonGetter{}, targetColumnLabel);
2259}
2260} // namespace row_helpers
2261} // namespace o2::soa
2262
2263namespace o2::aod
2264{
2266O2ORIGIN("AOD1");
2267O2ORIGIN("AOD2");
2270O2ORIGIN("JOIN");
2271O2HASH("JOIN/0");
2272O2ORIGIN("CONC");
2273O2HASH("CONC/0");
2274O2ORIGIN("TEST");
2275O2HASH("TEST/0");
2276} // namespace o2::aod
2277
2278#define DECLARE_EQUIVALENT_FOR_INDEX(_Base_, _Equiv_) \
2279 template <> \
2280 struct EquivalentIndexNG<o2::aod::Hash<_Base_::ref.desc_hash>, o2::aod::Hash<_Equiv_::ref.desc_hash>> { \
2281 constexpr static bool value = true; \
2282 }
2283
2284#define DECLARE_EQUIVALENT_FOR_INDEX_NG(_Base_, _Equiv_) \
2285 template <> \
2286 struct EquivalentIndexNG<o2::aod::Hash<_Base_ ""_h>, o2::aod::Hash<_Equiv_ ""_h>> { \
2287 constexpr static bool value = true; \
2288 }
2289
2290#define DECLARE_SOA_COLUMN_FULL(_Name_, _Getter_, _Type_, _Label_) \
2291 struct _Name_ : o2::soa::Column<_Type_, _Name_> { \
2292 static constexpr const char* mLabel = _Label_; \
2293 static_assert(!((*(mLabel + 1) == 'I' && *(mLabel + 2) == 'n' && *(mLabel + 3) == 'd' && *(mLabel + 4) == 'e' && *(mLabel + 5) == 'x')), "Index is not a valid column name"); \
2294 using base = o2::soa::Column<_Type_, _Name_>; \
2295 using type = _Type_; \
2296 using column_t = _Name_; \
2297 _Name_(arrow::ChunkedArray const* column) \
2298 : o2::soa::Column<_Type_, _Name_>(o2::soa::ColumnIterator<type>(column)) \
2299 { \
2300 } \
2301 \
2302 _Name_() = default; \
2303 _Name_(_Name_ const& other) = default; \
2304 _Name_& operator=(_Name_ const& other) = default; \
2305 \
2306 decltype(auto) _Getter_() const \
2307 { \
2308 return *mColumnIterator; \
2309 } \
2310 \
2311 decltype(auto) get() const \
2312 { \
2313 return _Getter_(); \
2314 } \
2315 }; \
2316 [[maybe_unused]] static constexpr o2::framework::expressions::BindingNode _Getter_ { _Label_, o2::framework::TypeIdHelpers::uniqueId<_Name_>(), \
2317 o2::framework::expressions::selectArrowType<_Type_>() }
2318
2319#define DECLARE_SOA_COLUMN(_Name_, _Getter_, _Type_) \
2320 DECLARE_SOA_COLUMN_FULL(_Name_, _Getter_, _Type_, "f" #_Name_)
2321
2324#define MAKEINT(_Size_) uint##_Size_##_t
2325
2326#define DECLARE_SOA_BITMAP_COLUMN_FULL(_Name_, _Getter_, _Size_, _Label_) \
2327 struct _Name_ : o2::soa::Column<MAKEINT(_Size_), _Name_> { \
2328 static constexpr const char* mLabel = _Label_; \
2329 static_assert(!((*(mLabel + 1) == 'I' && *(mLabel + 2) == 'n' && *(mLabel + 3) == 'd' && *(mLabel + 4) == 'e' && *(mLabel + 5) == 'x')), "Index is not a valid column name"); \
2330 using base = o2::soa::Column<MAKEINT(_Size_), _Name_>; \
2331 using type = MAKEINT(_Size_); \
2332 _Name_(arrow::ChunkedArray const* column) \
2333 : o2::soa::Column<type, _Name_>(o2::soa::ColumnIterator<type>(column)) \
2334 { \
2335 } \
2336 \
2337 _Name_() = default; \
2338 _Name_(_Name_ const& other) = default; \
2339 _Name_& operator=(_Name_ const& other) = default; \
2340 \
2341 decltype(auto) _Getter_##_raw() const \
2342 { \
2343 return *mColumnIterator; \
2344 } \
2345 \
2346 bool _Getter_##_bit(int bit) const \
2347 { \
2348 return (*mColumnIterator & (static_cast<type>(1) << bit)) >> bit; \
2349 } \
2350 }; \
2351 [[maybe_unused]] static constexpr o2::framework::expressions::BindingNode _Getter_ { _Label_, o2::framework::TypeIdHelpers::uniqueId<_Name_>(), \
2352 o2::framework::expressions::selectArrowType<MAKEINT(_Size_)>() }
2353
2354#define DECLARE_SOA_BITMAP_COLUMN(_Name_, _Getter_, _Size_) \
2355 DECLARE_SOA_BITMAP_COLUMN_FULL(_Name_, _Getter_, _Size_, "f" #_Name_)
2356
2359#define DECLARE_SOA_EXPRESSION_COLUMN_FULL(_Name_, _Getter_, _Type_, _Label_, _Expression_) \
2360 struct _Name_ : o2::soa::Column<_Type_, _Name_> { \
2361 static constexpr const char* mLabel = _Label_; \
2362 using base = o2::soa::Column<_Type_, _Name_>; \
2363 using type = _Type_; \
2364 using column_t = _Name_; \
2365 using spawnable_t = std::true_type; \
2366 _Name_(arrow::ChunkedArray const* column) \
2367 : o2::soa::Column<_Type_, _Name_>(o2::soa::ColumnIterator<type>(column)) \
2368 { \
2369 } \
2370 \
2371 _Name_() = default; \
2372 _Name_(_Name_ const& other) = default; \
2373 _Name_& operator=(_Name_ const& other) = default; \
2374 \
2375 decltype(auto) _Getter_() const \
2376 { \
2377 return *mColumnIterator; \
2378 } \
2379 \
2380 decltype(auto) get() const \
2381 { \
2382 return _Getter_(); \
2383 } \
2384 \
2385 static o2::framework::expressions::Projector Projector() \
2386 { \
2387 return _Expression_; \
2388 } \
2389 }; \
2390 [[maybe_unused]] static constexpr o2::framework::expressions::BindingNode _Getter_ { _Label_, o2::framework::TypeIdHelpers::uniqueId<_Name_>(), \
2391 o2::framework::expressions::selectArrowType<_Type_>() }
2392
2393#define DECLARE_SOA_EXPRESSION_COLUMN(_Name_, _Getter_, _Type_, _Expression_) \
2394 DECLARE_SOA_EXPRESSION_COLUMN_FULL(_Name_, _Getter_, _Type_, "f" #_Name_, _Expression_);
2395
2414
2416
2417template <o2::soa::is_table T>
2418consteval auto getIndexTargets()
2419{
2420 return T::originals;
2421}
2422
2423#define DECLARE_SOA_SLICE_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, _Label_, _Suffix_) \
2424 struct _Name_##IdSlice : o2::soa::Column<_Type_[2], _Name_##IdSlice> { \
2425 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2426 static_assert((*_Suffix_ == '\0') || (*_Suffix_ == '_'), "Suffix has to begin with _"); \
2427 static constexpr const char* mLabel = "fIndexSlice" _Label_ _Suffix_; \
2428 using base = o2::soa::Column<_Type_[2], _Name_##IdSlice>; \
2429 using type = _Type_[2]; \
2430 using column_t = _Name_##IdSlice; \
2431 using binding_t = _Table_; \
2432 static constexpr auto index_targets = getIndexTargets<_Table_>(); \
2433 _Name_##IdSlice(arrow::ChunkedArray const* column) \
2434 : o2::soa::Column<_Type_[2], _Name_##IdSlice>(o2::soa::ColumnIterator<type>(column)) \
2435 { \
2436 } \
2437 \
2438 _Name_##IdSlice() = default; \
2439 _Name_##IdSlice(_Name_##IdSlice const& other) = default; \
2440 _Name_##IdSlice& operator=(_Name_##IdSlice const& other) = default; \
2441 std::array<_Type_, 2> inline getIds() const \
2442 { \
2443 return _Getter_##Ids(); \
2444 } \
2445 \
2446 bool has_##_Getter_() const \
2447 { \
2448 auto a = *mColumnIterator; \
2449 return a[0] >= 0 && a[1] >= 0; \
2450 } \
2451 \
2452 std::array<_Type_, 2> _Getter_##Ids() const \
2453 { \
2454 auto a = *mColumnIterator; \
2455 return std::array{a[0], a[1]}; \
2456 } \
2457 \
2458 template <typename T> \
2459 auto _Getter_##_as() const \
2460 { \
2461 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2462 o2::soa::notBoundTable(#_Table_); \
2463 } \
2464 auto t = mBinding.get<T>(); \
2465 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2466 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2467 } \
2468 if (O2_BUILTIN_UNLIKELY(!has_##_Getter_())) { \
2469 return t->emptySlice(); \
2470 } \
2471 auto a = *mColumnIterator; \
2472 auto r = t->rawSlice(a[0], a[1]); \
2473 t->copyIndexBindings(r); \
2474 r.bindInternalIndicesTo(t); \
2475 return r; \
2476 } \
2477 \
2478 auto _Getter_() const \
2479 { \
2480 return _Getter_##_as<binding_t>(); \
2481 } \
2482 \
2483 template <typename T> \
2484 bool setCurrent(T const* current) \
2485 { \
2486 if constexpr (o2::soa::is_binding_compatible_v<T, binding_t>()) { \
2487 assert(current != nullptr); \
2488 this->mBinding.bind(current); \
2489 return true; \
2490 } \
2491 return false; \
2492 } \
2493 \
2494 bool setCurrentRaw(o2::soa::Binding current) \
2495 { \
2496 this->mBinding = current; \
2497 return true; \
2498 } \
2499 binding_t const* getCurrent() const { return mBinding.get<binding_t>(); } \
2500 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2501 o2::soa::Binding mBinding; \
2502 };
2503
2504#define DECLARE_SOA_SLICE_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Table_, _Suffix_) DECLARE_SOA_SLICE_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, #_Table_, _Suffix_)
2505#define DECLARE_SOA_SLICE_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_SLICE_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, _Name_##s, "")
2506#define DECLARE_SOA_SLICE_INDEX_COLUMN_CUSTOM(_Name_, _Getter_, _Label_) DECLARE_SOA_SLICE_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, int32_t, _Name_##s, _Label_, "")
2507
2509#define DECLARE_SOA_ARRAY_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, _Label_, _Suffix_) \
2510 struct _Name_##Ids : o2::soa::Column<std::vector<_Type_>, _Name_##Ids> { \
2511 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2512 static_assert((*_Suffix_ == '\0') || (*_Suffix_ == '_'), "Suffix has to begin with _"); \
2513 static constexpr const char* mLabel = "fIndexArray" _Label_ _Suffix_; \
2514 using base = o2::soa::Column<std::vector<_Type_>, _Name_##Ids>; \
2515 using type = std::vector<_Type_>; \
2516 using column_t = _Name_##Ids; \
2517 using binding_t = _Table_; \
2518 static constexpr auto index_targets = getIndexTargets<_Table_>(); \
2519 _Name_##Ids(arrow::ChunkedArray const* column) \
2520 : o2::soa::Column<std::vector<_Type_>, _Name_##Ids>(o2::soa::ColumnIterator<type>(column)) \
2521 { \
2522 } \
2523 \
2524 _Name_##Ids() = default; \
2525 _Name_##Ids(_Name_##Ids const& other) = default; \
2526 _Name_##Ids& operator=(_Name_##Ids const& other) = default; \
2527 \
2528 gsl::span<const _Type_> inline getIds() const \
2529 { \
2530 return _Getter_##Ids(); \
2531 } \
2532 \
2533 gsl::span<const _Type_> _Getter_##Ids() const \
2534 { \
2535 return *mColumnIterator; \
2536 } \
2537 \
2538 bool has_##_Getter_() const \
2539 { \
2540 return !(*mColumnIterator).empty(); \
2541 } \
2542 \
2543 template <typename T> \
2544 auto _Getter_##_as() const \
2545 { \
2546 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2547 o2::soa::notBoundTable(#_Table_); \
2548 } \
2549 auto t = mBinding.get<T>(); \
2550 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2551 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2552 } \
2553 return getIterators<T>(); \
2554 } \
2555 \
2556 template <typename T> \
2557 auto filtered_##_Getter_##_as() const \
2558 { \
2559 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2560 o2::soa::notBoundTable(#_Table_); \
2561 } \
2562 auto t = mBinding.get<T>(); \
2563 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2564 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2565 } \
2566 return getFilteredIterators<T>(); \
2567 } \
2568 \
2569 template <typename T> \
2570 auto getIterators() const \
2571 { \
2572 auto result = std::vector<typename T::unfiltered_iterator>(); \
2573 for (auto& i : *mColumnIterator) { \
2574 result.push_back(mBinding.get<T>()->rawIteratorAt(i)); \
2575 } \
2576 return result; \
2577 } \
2578 \
2579 template <typename T> \
2580 std::vector<typename T::iterator> getFilteredIterators() const \
2581 { \
2582 if constexpr (o2::soa::is_filtered_table<T>) { \
2583 auto result = std::vector<typename T::iterator>(); \
2584 for (auto const& i : *mColumnIterator) { \
2585 auto pos = mBinding.get<T>()->isInSelectedRows(i); \
2586 if (pos > 0) { \
2587 result.emplace_back(mBinding.get<T>()->iteratorAt(pos)); \
2588 } \
2589 } \
2590 return result; \
2591 } else { \
2592 static_assert(o2::framework::always_static_assert_v<T>, "T is not a Filtered type"); \
2593 } \
2594 return {}; \
2595 } \
2596 \
2597 auto _Getter_() const \
2598 { \
2599 return _Getter_##_as<binding_t>(); \
2600 } \
2601 \
2602 template <typename T> \
2603 auto _Getter_##_first_as() const \
2604 { \
2605 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2606 o2::soa::notBoundTable(#_Table_); \
2607 } \
2608 auto t = mBinding.get<T>(); \
2609 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2610 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2611 } \
2612 return t->rawIteratorAt((*mColumnIterator)[0]); \
2613 } \
2614 \
2615 template <typename T> \
2616 auto _Getter_##_last_as() const \
2617 { \
2618 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2619 o2::soa::notBoundTable(#_Table_); \
2620 } \
2621 auto t = mBinding.get<T>(); \
2622 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2623 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2624 } \
2625 return t->rawIteratorAt((*mColumnIterator).back()); \
2626 } \
2627 \
2628 auto _Getter_first() const \
2629 { \
2630 return _Getter_##_first_as<binding_t>(); \
2631 } \
2632 \
2633 auto _Getter_last() const \
2634 { \
2635 return _Getter_##_last_as<binding_t>(); \
2636 } \
2637 \
2638 template <typename T> \
2639 bool setCurrent(T const* current) \
2640 { \
2641 if constexpr (o2::soa::is_binding_compatible_v<T, binding_t>()) { \
2642 assert(current != nullptr); \
2643 this->mBinding.bind(current); \
2644 return true; \
2645 } \
2646 return false; \
2647 } \
2648 \
2649 bool setCurrentRaw(o2::soa::Binding current) \
2650 { \
2651 this->mBinding = current; \
2652 return true; \
2653 } \
2654 binding_t const* getCurrent() const { return mBinding.get<binding_t>(); } \
2655 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2656 o2::soa::Binding mBinding; \
2657 };
2658
2659#define DECLARE_SOA_ARRAY_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Table_, _Suffix_) DECLARE_SOA_ARRAY_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, #_Table_, _Suffix_)
2660#define DECLARE_SOA_ARRAY_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_ARRAY_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, _Name_##s, "")
2661#define DECLARE_SOA_ARRAY_INDEX_COLUMN_CUSTOM(_Name_, _Getter_, _Label_) DECLARE_SOA_ARRAY_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, int32_t, _Name_##s, _Label_, "")
2662
2664#define DECLARE_SOA_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, _Label_, _Suffix_) \
2665 struct _Name_##Id : o2::soa::Column<_Type_, _Name_##Id> { \
2666 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2667 static_assert((*_Suffix_ == '\0') || (*_Suffix_ == '_'), "Suffix has to begin with _"); \
2668 static constexpr const char* mLabel = "fIndex" _Label_ _Suffix_; \
2669 using base = o2::soa::Column<_Type_, _Name_##Id>; \
2670 using type = _Type_; \
2671 using column_t = _Name_##Id; \
2672 using binding_t = _Table_; \
2673 static constexpr auto index_targets = getIndexTargets<_Table_>(); \
2674 _Name_##Id(arrow::ChunkedArray const* column) \
2675 : o2::soa::Column<_Type_, _Name_##Id>(o2::soa::ColumnIterator<type>(column)) \
2676 { \
2677 } \
2678 \
2679 _Name_##Id() = default; \
2680 _Name_##Id(_Name_##Id const& other) = default; \
2681 _Name_##Id& operator=(_Name_##Id const& other) = default; \
2682 type inline getId() const \
2683 { \
2684 return _Getter_##Id(); \
2685 } \
2686 \
2687 type _Getter_##Id() const \
2688 { \
2689 return *mColumnIterator; \
2690 } \
2691 \
2692 bool has_##_Getter_() const \
2693 { \
2694 return *mColumnIterator >= 0; \
2695 } \
2696 \
2697 template <typename T> \
2698 auto _Getter_##_as() const \
2699 { \
2700 if (O2_BUILTIN_UNLIKELY(mBinding.ptr == nullptr)) { \
2701 o2::soa::notBoundTable(#_Table_); \
2702 } \
2703 if (O2_BUILTIN_UNLIKELY(!has_##_Getter_())) { \
2704 o2::soa::accessingInvalidIndexFor(#_Getter_); \
2705 } \
2706 auto t = mBinding.get<T>(); \
2707 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2708 o2::soa::dereferenceWithWrongType(#_Getter_, #_Table_); \
2709 } \
2710 return t->rawIteratorAt(*mColumnIterator); \
2711 } \
2712 \
2713 auto _Getter_() const \
2714 { \
2715 return _Getter_##_as<binding_t>(); \
2716 } \
2717 \
2718 template <typename T> \
2719 bool setCurrent(T* current) \
2720 { \
2721 if constexpr (o2::soa::is_binding_compatible_v<T, binding_t>()) { \
2722 assert(current != nullptr); \
2723 this->mBinding.bind(current); \
2724 return true; \
2725 } \
2726 return false; \
2727 } \
2728 \
2729 bool setCurrentRaw(o2::soa::Binding current) \
2730 { \
2731 this->mBinding = current; \
2732 return true; \
2733 } \
2734 binding_t const* getCurrent() const { return mBinding.get<binding_t>(); } \
2735 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2736 o2::soa::Binding mBinding; \
2737 }; \
2738 [[maybe_unused]] static constexpr o2::framework::expressions::BindingNode _Getter_##Id { "fIndex" #_Table_ _Suffix_, o2::framework::TypeIdHelpers::uniqueId<_Name_##Id>(), \
2739 o2::framework::expressions::selectArrowType<_Type_>() }
2740
2741#define DECLARE_SOA_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Table_, _Suffix_) DECLARE_SOA_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, _Type_, _Table_, #_Table_, _Suffix_)
2742#define DECLARE_SOA_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, _Name_##s, "")
2743#define DECLARE_SOA_INDEX_COLUMN_CUSTOM(_Name_, _Getter_, _Label_) DECLARE_SOA_INDEX_COLUMN_FULL_CUSTOM(_Name_, _Getter_, int32_t, _Name_##s, _Label_, "")
2744
2746#define DECLARE_SOA_SELF_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, _IndexTarget_) \
2747 struct _Name_##Id : o2::soa::Column<_Type_, _Name_##Id> { \
2748 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2749 static constexpr const char* mLabel = "fIndex" _Label_; \
2750 using base = o2::soa::Column<_Type_, _Name_##Id>; \
2751 using type = _Type_; \
2752 using column_t = _Name_##Id; \
2753 using self_index_t = std::true_type; \
2754 using compatible_signature = std::conditional<aod::is_aod_hash<_IndexTarget_>, _IndexTarget_, void>; \
2755 _Name_##Id(arrow::ChunkedArray const* column) \
2756 : o2::soa::Column<_Type_, _Name_##Id>(o2::soa::ColumnIterator<type>(column)) \
2757 { \
2758 } \
2759 \
2760 _Name_##Id() = default; \
2761 _Name_##Id(_Name_##Id const& other) = default; \
2762 _Name_##Id& operator=(_Name_##Id const& other) = default; \
2763 type inline getId() const \
2764 { \
2765 return _Getter_##Id(); \
2766 } \
2767 \
2768 type _Getter_##Id() const \
2769 { \
2770 return *mColumnIterator; \
2771 } \
2772 \
2773 bool has_##_Getter_() const \
2774 { \
2775 return *mColumnIterator >= 0; \
2776 } \
2777 \
2778 template <typename T> \
2779 auto _Getter_##_as() const \
2780 { \
2781 if (O2_BUILTIN_UNLIKELY(!has_##_Getter_())) { \
2782 o2::soa::accessingInvalidIndexFor(#_Getter_); \
2783 } \
2784 auto t = mBinding.get<T>(); \
2785 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2786 o2::soa::dereferenceWithWrongType(#_Getter_, "self"); \
2787 } \
2788 return t->rawIteratorAt(*mColumnIterator); \
2789 } \
2790 \
2791 bool setCurrentRaw(o2::soa::Binding current) \
2792 { \
2793 this->mBinding = current; \
2794 return true; \
2795 } \
2796 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2797 o2::soa::Binding mBinding; \
2798 }; \
2799 [[maybe_unused]] static constexpr o2::framework::expressions::BindingNode _Getter_##Id { "fIndex" _Label_, o2::framework::TypeIdHelpers::uniqueId<_Name_##Id>(), \
2800 o2::framework::expressions::selectArrowType<_Type_>() }
2801
2802#define DECLARE_SOA_SELF_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Label_) DECLARE_SOA_SELF_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, void)
2803#define DECLARE_SOA_SELF_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_SELF_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, #_Name_)
2805#define DECLARE_SOA_SELF_SLICE_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, _IndexTarget_) \
2806 struct _Name_##IdSlice : o2::soa::Column<_Type_[2], _Name_##IdSlice> { \
2807 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2808 static constexpr const char* mLabel = "fIndexSlice" _Label_; \
2809 using base = o2::soa::Column<_Type_[2], _Name_##IdSlice>; \
2810 using type = _Type_[2]; \
2811 using column_t = _Name_##IdSlice; \
2812 using self_index_t = std::true_type; \
2813 using compatible_signature = std::conditional<aod::is_aod_hash<_IndexTarget_>, _IndexTarget_, void>; \
2814 _Name_##IdSlice(arrow::ChunkedArray const* column) \
2815 : o2::soa::Column<_Type_[2], _Name_##IdSlice>(o2::soa::ColumnIterator<type>(column)) \
2816 { \
2817 } \
2818 \
2819 _Name_##IdSlice() = default; \
2820 _Name_##IdSlice(_Name_##IdSlice const& other) = default; \
2821 _Name_##IdSlice& operator=(_Name_##IdSlice const& other) = default; \
2822 std::array<_Type_, 2> inline getIds() const \
2823 { \
2824 return _Getter_##Ids(); \
2825 } \
2826 \
2827 bool has_##_Getter_() const \
2828 { \
2829 auto a = *mColumnIterator; \
2830 return a[0] >= 0 && a[1] >= 0; \
2831 } \
2832 \
2833 std::array<_Type_, 2> _Getter_##Ids() const \
2834 { \
2835 auto a = *mColumnIterator; \
2836 return std::array{a[0], a[1]}; \
2837 } \
2838 \
2839 template <typename T> \
2840 auto _Getter_##_as() const \
2841 { \
2842 auto t = mBinding.get<T>(); \
2843 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2844 o2::soa::dereferenceWithWrongType(#_Getter_, "self"); \
2845 } \
2846 if (O2_BUILTIN_UNLIKELY(!has_##_Getter_())) { \
2847 return t->emptySlice(); \
2848 } \
2849 auto a = *mColumnIterator; \
2850 auto r = t->rawSlice(a[0], a[1]); \
2851 t->copyIndexBindings(r); \
2852 r.bindInternalIndicesTo(t); \
2853 return r; \
2854 } \
2855 \
2856 bool setCurrentRaw(o2::soa::Binding current) \
2857 { \
2858 this->mBinding = current; \
2859 return true; \
2860 } \
2861 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2862 o2::soa::Binding mBinding; \
2863 };
2864
2865#define DECLARE_SOA_SELF_SLICE_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Label_) DECLARE_SOA_SELF_SLICE_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, void)
2866#define DECLARE_SOA_SELF_SLICE_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_SELF_SLICE_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, "_" #_Name_)
2868#define DECLARE_SOA_SELF_ARRAY_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, _IndexTarget_) \
2869 struct _Name_##Ids : o2::soa::Column<std::vector<_Type_>, _Name_##Ids> { \
2870 static_assert(std::is_integral_v<_Type_>, "Index type must be integral"); \
2871 static constexpr const char* mLabel = "fIndexArray" _Label_; \
2872 using base = o2::soa::Column<std::vector<_Type_>, _Name_##Ids>; \
2873 using type = std::vector<_Type_>; \
2874 using column_t = _Name_##Ids; \
2875 using self_index_t = std::true_type; \
2876 using compatible_signature = std::conditional<aod::is_aod_hash<_IndexTarget_>, _IndexTarget_, void>; \
2877 _Name_##Ids(arrow::ChunkedArray const* column) \
2878 : o2::soa::Column<std::vector<_Type_>, _Name_##Ids>(o2::soa::ColumnIterator<type>(column)) \
2879 { \
2880 } \
2881 \
2882 _Name_##Ids() = default; \
2883 _Name_##Ids(_Name_##Ids const& other) = default; \
2884 _Name_##Ids& operator=(_Name_##Ids const& other) = default; \
2885 gsl::span<const _Type_> inline getIds() const \
2886 { \
2887 return _Getter_##Ids(); \
2888 } \
2889 \
2890 gsl::span<const _Type_> _Getter_##Ids() const \
2891 { \
2892 return *mColumnIterator; \
2893 } \
2894 \
2895 bool has_##_Getter_() const \
2896 { \
2897 return !(*mColumnIterator).empty(); \
2898 } \
2899 \
2900 template <typename T> \
2901 auto _Getter_##_as() const \
2902 { \
2903 auto t = mBinding.get<T>(); \
2904 if (O2_BUILTIN_UNLIKELY(t == nullptr)) { \
2905 o2::soa::dereferenceWithWrongType(#_Getter_, "self"); \
2906 } \
2907 return getIterators<T>(); \
2908 } \
2909 \
2910 template <typename T> \
2911 auto getIterators() const \
2912 { \
2913 auto result = std::vector<typename T::unfiltered_iterator>(); \
2914 for (auto& i : *mColumnIterator) { \
2915 result.push_back(mBinding.get<T>()->rawIteratorAt(i)); \
2916 } \
2917 return result; \
2918 } \
2919 \
2920 template <typename T> \
2921 auto _Getter_##_first_as() const \
2922 { \
2923 return mBinding.get<T>()->rawIteratorAt((*mColumnIterator)[0]); \
2924 } \
2925 \
2926 template <typename T> \
2927 auto _Getter_##_last_as() const \
2928 { \
2929 return mBinding.get<T>()->rawIteratorAt((*mColumnIterator).back()); \
2930 } \
2931 \
2932 bool setCurrentRaw(o2::soa::Binding current) \
2933 { \
2934 this->mBinding = current; \
2935 return true; \
2936 } \
2937 o2::soa::Binding getCurrentRaw() const { return mBinding; } \
2938 o2::soa::Binding mBinding; \
2939 };
2940
2941#define DECLARE_SOA_SELF_ARRAY_INDEX_COLUMN_FULL(_Name_, _Getter_, _Type_, _Label_) DECLARE_SOA_SELF_ARRAY_INDEX_COLUMN_COMPLETE(_Name_, _Getter_, _Type_, _Label_, void)
2942#define DECLARE_SOA_SELF_ARRAY_INDEX_COLUMN(_Name_, _Getter_) DECLARE_SOA_SELF_ARRAY_INDEX_COLUMN_FULL(_Name_, _Getter_, int32_t, "_" #_Name_)
2943
2972#define DECLARE_SOA_DYNAMIC_COLUMN(_Name_, _Getter_, ...) \
2973 struct _Name_##Callback { \
2974 static inline constexpr auto getLambda() { return __VA_ARGS__; } \
2975 }; \
2976 \
2977 struct _Name_##Helper { \
2978 using callable_t = decltype(o2::framework::FunctionMetadata(std::declval<decltype(_Name_##Callback::getLambda())>())); \
2979 using return_type = typename callable_t::return_type; \
2980 }; \
2981 template <typename... Bindings> \
2982 struct _Name_ : o2::soa::DynamicColumn<typename _Name_##Helper::callable_t::type, _Name_<Bindings...>> { \
2983 using base = o2::soa::DynamicColumn<typename _Name_##Helper::callable_t::type, _Name_<Bindings...>>; \
2984 using helper = _Name_##Helper; \
2985 using callback_holder_t = _Name_##Callback; \
2986 using callable_t = helper::callable_t; \
2987 using callback_t = callable_t::type; \
2988 \
2989 _Name_(arrow::ChunkedArray const*) \
2990 { \
2991 } \
2992 _Name_() = default; \
2993 _Name_(_Name_ const& other) = default; \
2994 _Name_& operator=(_Name_ const& other) = default; \
2995 static constexpr const char* mLabel = #_Name_; \
2996 using type = typename callable_t::return_type; \
2997 \
2998 template <typename... FreeArgs> \
2999 type _Getter_(FreeArgs... freeArgs) const \
3000 { \
3001 return boundGetter(std::make_index_sequence<std::tuple_size_v<decltype(boundIterators)>>{}, freeArgs...); \
3002 } \
3003 template <typename... FreeArgs> \
3004 type getDynamicValue(FreeArgs... freeArgs) const \
3005 { \
3006 return boundGetter(std::make_index_sequence<std::tuple_size_v<decltype(boundIterators)>>{}, freeArgs...); \
3007 } \
3008 \
3009 type get() const \
3010 { \
3011 return _Getter_(); \
3012 } \
3013 \
3014 template <size_t... Is, typename... FreeArgs> \
3015 type boundGetter(std::integer_sequence<size_t, Is...>&&, FreeArgs... freeArgs) const \
3016 { \
3017 return __VA_ARGS__((**std::get<Is>(boundIterators))..., freeArgs...); \
3018 } \
3019 \
3020 using bindings_t = typename o2::framework::pack<Bindings...>; \
3021 std::tuple<o2::soa::ColumnIterator<typename Bindings::type> const*...> boundIterators; \
3022 }
3023
3024#define DECLARE_SOA_TABLE_METADATA(_Name_, _Desc_, _Version_, ...) \
3025 using _Name_##Metadata = TableMetadata<Hash<_Desc_ "/" #_Version_ ""_h>, __VA_ARGS__>;
3026
3027#define DECLARE_SOA_TABLE_METADATA_TRAIT(_Name_, _Desc_, _Version_) \
3028 template <> \
3029 struct MetadataTrait<Hash<_Desc_ "/" #_Version_ ""_h>> { \
3030 using metadata = _Name_##Metadata; \
3031 };
3032
3033#define DECLARE_SOA_TABLE_FULL_VERSIONED_(_Name_, _Label_, _Origin_, _Desc_, _Version_) \
3034 O2HASH(_Desc_ "/" #_Version_); \
3035 template <typename O> \
3036 using _Name_##From = o2::soa::Table<Hash<_Label_ ""_h>, Hash<_Desc_ "/" #_Version_ ""_h>, O>; \
3037 using _Name_ = _Name_##From<Hash<_Origin_ ""_h>>; \
3038 template <> \
3039 struct MetadataTrait<Hash<_Desc_ "/" #_Version_ ""_h>> { \
3040 using metadata = _Name_##Metadata; \
3041 };
3042
3043#define DECLARE_SOA_STAGE(_Name_, _Origin_, _Desc_, _Version_) \
3044 template <typename O> \
3045 using _Name_##From = o2::soa::Table<Hash<#_Name_ ""_h>, Hash<_Desc_ "/" #_Version_ ""_h>, O>; \
3046 using _Name_ = _Name_##From<Hash<_Origin_ ""_h>>;
3047
3048#define DECLARE_SOA_TABLE_FULL_VERSIONED(_Name_, _Label_, _Origin_, _Desc_, _Version_, ...) \
3049 DECLARE_SOA_TABLE_METADATA(_Name_, _Desc_, _Version_, __VA_ARGS__); \
3050 DECLARE_SOA_TABLE_FULL_VERSIONED_(_Name_, _Label_, _Origin_, _Desc_, _Version_);
3051
3052#define DECLARE_SOA_TABLE_FULL(_Name_, _Label_, _Origin_, _Desc_, ...) \
3053 O2HASH(_Label_); \
3054 DECLARE_SOA_TABLE_METADATA(_Name_, _Desc_, 0, __VA_ARGS__); \
3055 DECLARE_SOA_TABLE_FULL_VERSIONED_(_Name_, _Label_, _Origin_, _Desc_, 0)
3056
3057#define DECLARE_SOA_TABLE(_Name_, _Origin_, _Desc_, ...) \
3058 DECLARE_SOA_TABLE_FULL(_Name_, #_Name_, _Origin_, _Desc_, __VA_ARGS__)
3059
3060#define DECLARE_SOA_TABLE_VERSIONED(_Name_, _Origin_, _Desc_, _Version_, ...) \
3061 O2HASH(#_Name_); \
3062 DECLARE_SOA_TABLE_METADATA(_Name_, _Desc_, _Version_, __VA_ARGS__); \
3063 DECLARE_SOA_TABLE_FULL_VERSIONED_(_Name_, #_Name_, _Origin_, _Desc_, _Version_)
3064
3065#define DECLARE_SOA_TABLE_STAGED_VERSIONED(_BaseName_, _Desc_, _Version_, ...) \
3066 O2HASH(_Desc_ "/" #_Version_); \
3067 O2HASH(#_BaseName_); \
3068 O2HASH("Stored" #_BaseName_); \
3069 DECLARE_SOA_TABLE_METADATA(_BaseName_, _Desc_, _Version_, __VA_ARGS__); \
3070 using Stored##_BaseName_##Metadata = _BaseName_##Metadata; \
3071 DECLARE_SOA_TABLE_METADATA_TRAIT(_BaseName_, _Desc_, _Version_); \
3072 DECLARE_SOA_STAGE(_BaseName_, "AOD", _Desc_, _Version_); \
3073 DECLARE_SOA_STAGE(Stored##_BaseName_, "AOD1", _Desc_, _Version_);
3074
3075#define DECLARE_SOA_TABLE_STAGED(_BaseName_, _Desc_, ...) \
3076 DECLARE_SOA_TABLE_STAGED_VERSIONED(_BaseName_, _Desc_, 0, __VA_ARGS__);
3077
3078#define DECLARE_SOA_EXTENDED_TABLE_FULL(_Name_, _Label_, _OriginalTable_, _Origin_, _Desc_, _Version_, ...) \
3079 O2HASH(_Desc_ "/" #_Version_); \
3080 template <typename O> \
3081 using _Name_##ExtensionFrom = soa::Table<o2::aod::Hash<_Label_ ""_h>, o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>, O>; \
3082 using _Name_##Extension = _Name_##ExtensionFrom<o2::aod::Hash<_Origin_ ""_h>>; \
3083 template <typename O = o2::aod::Hash<_Origin_ ""_h>> \
3084 struct _Name_##ExtensionMetadataFrom : TableMetadata<o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>, __VA_ARGS__> { \
3085 using base_table_t = _OriginalTable_; \
3086 using extension_table_t = _Name_##ExtensionFrom<O>; \
3087 using expression_pack_t = framework::pack<__VA_ARGS__>; \
3088 static constexpr auto sources = _OriginalTable_::originals; \
3089 }; \
3090 using _Name_##ExtensionMetadata = _Name_##ExtensionMetadataFrom<o2::aod::Hash<_Origin_ ""_h>>; \
3091 template <> \
3092 struct MetadataTrait<o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>> { \
3093 using metadata = _Name_##ExtensionMetadata; \
3094 }; \
3095 template <typename O> \
3096 using _Name_##From = o2::soa::JoinFull<o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>, _OriginalTable_, _Name_##ExtensionFrom<O>>; \
3097 using _Name_ = _Name_##From<o2::aod::Hash<_Origin_ ""_h>>;
3098
3099#define DECLARE_SOA_EXTENDED_TABLE(_Name_, _Table_, _Description_, _Version_, ...) \
3100 O2HASH(#_Name_ "Extension"); \
3101 DECLARE_SOA_EXTENDED_TABLE_FULL(_Name_, #_Name_ "Extension", _Table_, "DYN", _Description_, _Version_, __VA_ARGS__)
3102
3103#define DECLARE_SOA_EXTENDED_TABLE_USER(_Name_, _Table_, _Description_, ...) \
3104 O2HASH(#_Name_ "Extension"); \
3105 DECLARE_SOA_EXTENDED_TABLE_FULL(_Name_, #_Name_ "Extension", _Table_, "AOD", "EX" _Description_, 0, __VA_ARGS__)
3106
3107#define DECLARE_SOA_INDEX_TABLE_FULL(_Name_, _Key_, _Origin_, _Version_, _Desc_, _Exclusive_, ...) \
3108 O2HASH(#_Name_); \
3109 O2HASH(_Desc_ "/" #_Version_); \
3110 template <typename O = o2::aod::Hash<_Origin_ ""_h>> \
3111 struct _Name_##MetadataFrom : o2::aod::TableMetadata<o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>, soa::Index<>, __VA_ARGS__> { \
3112 static constexpr bool exclusive = _Exclusive_; \
3113 using Key = _Key_; \
3114 using index_pack_t = framework::pack<__VA_ARGS__>; \
3115 static constexpr const auto sources = []<typename... Cs>(framework::pack<Cs...>) { \
3116 constexpr auto a = o2::soa::mergeOriginals<typename Cs::binding_t...>(); \
3117 return o2::aod::filterForKey<a.size(), a, Key>(); \
3118 }(framework::pack<__VA_ARGS__>{}); \
3119 }; \
3120 using _Name_##Metadata = _Name_##MetadataFrom<o2::aod::Hash<_Origin_ ""_h>>; \
3121 \
3122 template <typename O = o2::aod::Hash<_Origin_ ""_h>> \
3123 using _Name_##From = o2::soa::IndexTable<o2::aod::Hash<#_Name_ ""_h>, o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>, O, _Key_, __VA_ARGS__>; \
3124 using _Name_ = _Name_##From<o2::aod::Hash<_Origin_ ""_h>>; \
3125 \
3126 template <> \
3127 struct MetadataTrait<o2::aod::Hash<_Desc_ "/" #_Version_ ""_h>> { \
3128 using metadata = _Name_##Metadata; \
3129 };
3130
3131#define DECLARE_SOA_INDEX_TABLE(_Name_, _Key_, _Description_, ...) \
3132 DECLARE_SOA_INDEX_TABLE_FULL(_Name_, _Key_, "IDX", 0, _Description_, false, __VA_ARGS__)
3133
3134#define DECLARE_SOA_INDEX_TABLE_EXCLUSIVE(_Name_, _Key_, _Description_, ...) \
3135 DECLARE_SOA_INDEX_TABLE_FULL(_Name_, _Key_, "IDX", 0, _Description_, true, __VA_ARGS__)
3136
3137#define DECLARE_SOA_INDEX_TABLE_USER(_Name_, _Key_, _Description_, ...) \
3138 DECLARE_SOA_INDEX_TABLE_FULL(_Name_, _Key_, "AOD", 0, _Description_, false, __VA_ARGS__)
3139
3140#define DECLARE_SOA_INDEX_TABLE_EXCLUSIVE_USER(_Name_, _Key_, _Description_, ...) \
3141 DECLARE_SOA_INDEX_TABLE_FULL(_Name_, _Key_, "AOD", 0, _Description_, true, __VA_ARGS__)
3142
3143namespace o2::soa
3144{
3145template <typename D, typename... Ts>
3146struct JoinFull : Table<o2::aod::Hash<"JOIN"_h>, D, o2::aod::Hash<"JOIN"_h>, Ts...> {
3147 using base = Table<o2::aod::Hash<"JOIN"_h>, D, o2::aod::Hash<"JOIN"_h>, Ts...>;
3148
3149 JoinFull(std::shared_ptr<arrow::Table>&& table, uint64_t offset = 0)
3150 : base{std::move(table), offset}
3151 {
3153 }
3154 JoinFull(std::vector<std::shared_ptr<arrow::Table>>&& tables, uint64_t offset = 0)
3155 : base{ArrowHelpers::joinTables(std::move(tables)), offset}
3156 {
3158 }
3161
3162 using self_t = JoinFull<D, Ts...>;
3163 using table_t = base;
3164 static constexpr const auto originals = base::originals;
3167 using iterator = table_t::template iterator_template<DefaultIndexPolicy, self_t, Ts...>;
3173
3175 {
3176 return iterator{this->cached_begin()};
3177 }
3178
3180 {
3181 return const_iterator{this->cached_begin()};
3182 }
3183
3185 {
3186 return doSliceByCached(this, node, value, cache);
3187 }
3188
3190 {
3191 return doSliceByCachedUnsorted(this, node, value, cache);
3192 }
3193
3194 template <typename T1, typename Policy, bool OPT>
3196 {
3197 return doSliceBy(this, container, value);
3198 }
3199
3200 iterator rawIteratorAt(uint64_t i) const
3201 {
3202 auto it = iterator{this->cached_begin()};
3203 it.setCursor(i);
3204 return it;
3205 }
3206
3207 iterator iteratorAt(uint64_t i) const
3208 {
3209 return rawIteratorAt(i);
3210 }
3211
3212 auto rawSlice(uint64_t start, uint64_t end) const
3213 {
3214 return self_t{{this->asArrowTable()->Slice(start, end - start + 1)}, start};
3215 }
3216
3217 auto emptySlice() const
3218 {
3219 return self_t{{this->asArrowTable()->Slice(0, 0)}, 0};
3220 }
3221
3222 template <typename T>
3223 static consteval bool contains()
3224 {
3225 return std::find_if(originals.begin(), originals.end(), [](TableRef const& ref) { return ref.desc_hash == T::ref.desc_hash; }) != originals.end();
3226 }
3227};
3228
3229template <typename... Ts>
3230using Join = JoinFull<o2::aod::Hash<"JOIN/0"_h>, Ts...>;
3231
3232template <typename... Ts>
3233constexpr auto join(Ts const&... t)
3234{
3235 return Join<Ts...>(ArrowHelpers::joinTables({t.asArrowTable()...}));
3236}
3237
3238template <typename T>
3240
3241template <typename T>
3242constexpr bool is_soa_join_v = is_join<T>;
3243
3244template <typename... Ts>
3245struct Concat : Table<o2::aod::Hash<"CONC"_h>, o2::aod::Hash<"CONC/0"_h>, o2::aod::Hash<"CONC"_h>, Ts...> {
3246 using base = Table<o2::aod::Hash<"CONC"_h>, o2::aod::Hash<"CONC/0"_h>, o2::aod::Hash<"CONC"_h>, Ts...>;
3247 using self_t = Concat<Ts...>;
3248 Concat(std::vector<std::shared_ptr<arrow::Table>>&& tables, uint64_t offset = 0)
3249 : base{ArrowHelpers::concatTables(std::move(tables)), offset}
3250 {
3252 }
3253 Concat(Ts const&... t, uint64_t offset = 0)
3254 : base{ArrowHelpers::concatTables({t.asArrowTable()...}), offset}
3255 {
3256 bindInternalIndicesTo(this);
3257 }
3258
3259 using base::originals;
3260
3261 using base::bindExternalIndices;
3262 using base::bindInternalIndicesTo;
3263
3264 using table_t = base;
3267
3268 using iterator = table_t::template iterator_template<DefaultIndexPolicy, self_t, Ts...>;
3274};
3275
3276template <typename... Ts>
3277constexpr auto concat(Ts const&... t)
3278{
3279 return Concat<Ts...>{t...};
3280}
3281
3282template <soa::is_table T>
3283class FilteredBase : public T
3284{
3285 public:
3287 using table_t = typename T::table_t;
3288 using T::originals;
3289 using columns_t = typename T::columns_t;
3290 using persistent_columns_t = typename T::persistent_columns_t;
3291 using external_index_columns_t = typename T::external_index_columns_t;
3292
3293 using iterator = T::template iterator_template_o<FilteredIndexPolicy, self_t>;
3294 using unfiltered_iterator = T::template iterator_template_o<DefaultIndexPolicy, self_t>;
3296
3297 FilteredBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, gandiva::Selection const& selection, uint64_t offset = 0)
3298 : T{std::move(tables), offset},
3299 mSelectedRows{getSpan(selection)}
3300 {
3301 if (this->tableSize() != 0) {
3302 mFilteredBegin = table_t::filtered_begin(mSelectedRows);
3303 }
3304 resetRanges();
3305 mFilteredBegin.bindInternalIndices(this);
3306 }
3307
3308 FilteredBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, SelectionVector&& selection, uint64_t offset = 0)
3309 : T{std::move(tables), offset},
3310 mSelectedRowsCache{std::move(selection)},
3311 mCached{true}
3312 {
3313 mSelectedRows = gsl::span{mSelectedRowsCache};
3314 if (this->tableSize() != 0) {
3315 mFilteredBegin = table_t::filtered_begin(mSelectedRows);
3316 }
3317 resetRanges();
3318 mFilteredBegin.bindInternalIndices(this);
3319 }
3320
3321 FilteredBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, gsl::span<int64_t const> const& selection, uint64_t offset = 0)
3322 : T{std::move(tables), offset},
3323 mSelectedRows{selection}
3324 {
3325 if (this->tableSize() != 0) {
3326 mFilteredBegin = table_t::filtered_begin(mSelectedRows);
3327 }
3328 resetRanges();
3329 mFilteredBegin.bindInternalIndices(this);
3330 }
3331
3333 {
3334 return iterator(mFilteredBegin);
3335 }
3336
3338 {
3339 return const_iterator(mFilteredBegin);
3340 }
3341
3343 {
3344 auto it = unfiltered_iterator{mFilteredBegin};
3345 it.setCursor(i);
3346 return it;
3347 }
3348
3349 [[nodiscard]] RowViewSentinel end() const
3350 {
3351 return RowViewSentinel{*mFilteredEnd};
3352 }
3353
3355 {
3356 return mFilteredBegin;
3357 }
3358
3359 auto const& cached_begin() const
3360 {
3361 return mFilteredBegin;
3362 }
3363
3364 iterator iteratorAt(uint64_t i) const
3365 {
3366 return mFilteredBegin + i;
3367 }
3368
3369 [[nodiscard]] int64_t size() const
3370 {
3371 return mSelectedRows.size();
3372 }
3373
3374 [[nodiscard]] int64_t tableSize() const
3375 {
3376 return table_t::asArrowTable()->num_rows();
3377 }
3378
3379 auto const& getSelectedRows() const
3380 {
3381 return mSelectedRows;
3382 }
3383
3384 auto rawSlice(uint64_t start, uint64_t end) const
3385 {
3386 SelectionVector newSelection;
3387 newSelection.resize(static_cast<int64_t>(end - start + 1));
3388 std::iota(newSelection.begin(), newSelection.end(), start);
3389 return self_t{{this->asArrowTable()}, std::move(newSelection), 0};
3390 }
3391
3392 auto emptySlice() const
3393 {
3394 return self_t{{this->asArrowTable()}, SelectionVector{}, 0};
3395 }
3396
3397 static inline auto getSpan(gandiva::Selection const& sel)
3398 {
3399 if (sel == nullptr) {
3400 return gsl::span<int64_t const>{};
3401 }
3402 auto array = std::static_pointer_cast<arrow::Int64Array>(sel->ToArray());
3403 auto start = array->raw_values();
3404 auto stop = start + array->length();
3405 return gsl::span{start, stop};
3406 }
3407
3410 template <typename... TA>
3411 void bindExternalIndices(TA*... current)
3412 {
3413 table_t::bindExternalIndices(current...);
3414 mFilteredBegin.bindExternalIndices(current...);
3415 }
3416
3417 void bindExternalIndicesRaw(std::vector<o2::soa::Binding>&& ptrs)
3418 {
3419 mFilteredBegin.bindExternalIndicesRaw(std::forward<std::vector<o2::soa::Binding>>(ptrs));
3420 }
3421
3422 template <typename I>
3424 {
3425 mFilteredBegin.bindInternalIndices(ptr);
3426 }
3427
3428 template <typename T1, typename... Cs>
3430 {
3431 dest.bindExternalIndicesRaw(mFilteredBegin.getIndexBindings());
3432 }
3433
3434 template <typename T1>
3435 void copyIndexBindings(T1& dest) const
3436 {
3437 doCopyIndexBindings(external_index_columns_t{}, dest);
3438 }
3439
3440 template <typename T1>
3441 auto rawSliceBy(o2::framework::Preslice<T1> const& container, int value) const
3442 {
3443 return (table_t)this->sliceBy(container, value);
3444 }
3445
3447 {
3448 return doFilteredSliceByCached(this, node, value, cache);
3449 }
3450
3452 {
3453 return doSliceByCachedUnsorted(this, node, value, cache);
3454 }
3455
3456 template <typename T1, bool OPT>
3458 {
3459 return doFilteredSliceBy(this, container, value);
3460 }
3461
3462 template <typename T1, bool OPT>
3464 {
3465 return doSliceBy(this, container, value);
3466 }
3467
3469 {
3470 auto t = o2::soa::select(*this, f);
3471 copyIndexBindings(t);
3472 return t;
3473 }
3474
3475 int isInSelectedRows(int i) const
3476 {
3477 auto locate = std::find(mSelectedRows.begin(), mSelectedRows.end(), i);
3478 if (locate == mSelectedRows.end()) {
3479 return -1;
3480 }
3481 return static_cast<int>(std::distance(mSelectedRows.begin(), locate));
3482 }
3483
3484 void sumWithSelection(SelectionVector const& selection)
3485 {
3486 mCached = true;
3487 SelectionVector rowsUnion;
3488 std::set_union(mSelectedRows.begin(), mSelectedRows.end(), selection.begin(), selection.end(), std::back_inserter(rowsUnion));
3489 mSelectedRowsCache.clear();
3490 mSelectedRowsCache = rowsUnion;
3491 resetRanges();
3492 }
3493
3495 {
3496 mCached = true;
3497 SelectionVector intersection;
3498 std::set_intersection(mSelectedRows.begin(), mSelectedRows.end(), selection.begin(), selection.end(), std::back_inserter(intersection));
3499 mSelectedRowsCache.clear();
3500 mSelectedRowsCache = intersection;
3501 resetRanges();
3502 }
3503
3504 void sumWithSelection(gsl::span<int64_t const> const& selection)
3505 {
3506 mCached = true;
3507 SelectionVector rowsUnion;
3508 std::set_union(mSelectedRows.begin(), mSelectedRows.end(), selection.begin(), selection.end(), std::back_inserter(rowsUnion));
3509 mSelectedRowsCache.clear();
3510 mSelectedRowsCache = rowsUnion;
3511 resetRanges();
3512 }
3513
3514 void intersectWithSelection(gsl::span<int64_t const> const& selection)
3515 {
3516 mCached = true;
3517 SelectionVector intersection;
3518 std::set_intersection(mSelectedRows.begin(), mSelectedRows.end(), selection.begin(), selection.end(), std::back_inserter(intersection));
3519 mSelectedRowsCache.clear();
3520 mSelectedRowsCache = intersection;
3521 resetRanges();
3522 }
3523
3524 bool isCached() const
3525 {
3526 return mCached;
3527 }
3528
3529 private:
3530 void resetRanges()
3531 {
3532 if (mCached) {
3533 mSelectedRows = gsl::span{mSelectedRowsCache};
3534 }
3535 mFilteredEnd.reset(new RowViewSentinel{static_cast<int64_t>(mSelectedRows.size())});
3536 if (tableSize() == 0) {
3537 mFilteredBegin = *mFilteredEnd;
3538 } else {
3539 mFilteredBegin.resetSelection(mSelectedRows);
3540 }
3541 }
3542
3543 gsl::span<int64_t const> mSelectedRows;
3544 SelectionVector mSelectedRowsCache;
3545 bool mCached = false;
3546 iterator mFilteredBegin;
3547 std::shared_ptr<RowViewSentinel> mFilteredEnd;
3548};
3549
3550template <typename T>
3551class Filtered : public FilteredBase<T>
3552{
3553 public:
3554 using base_t = T;
3556 using table_t = typename T::table_t;
3557 using columns_t = typename T::columns_t;
3558
3559 using iterator = T::template iterator_template_o<FilteredIndexPolicy, self_t>;
3560 using unfiltered_iterator = T::template iterator_template_o<DefaultIndexPolicy, self_t>;
3562
3564 {
3565 return iterator(this->cached_begin());
3566 }
3567
3569 {
3570 return const_iterator(this->cached_begin());
3571 }
3572
3573 Filtered(std::vector<std::shared_ptr<arrow::Table>>&& tables, gandiva::Selection const& selection, uint64_t offset = 0)
3574 : FilteredBase<T>(std::move(tables), selection, offset) {}
3575
3576 Filtered(std::vector<std::shared_ptr<arrow::Table>>&& tables, SelectionVector&& selection, uint64_t offset = 0)
3577 : FilteredBase<T>(std::move(tables), std::forward<SelectionVector>(selection), offset) {}
3578
3579 Filtered(std::vector<std::shared_ptr<arrow::Table>>&& tables, gsl::span<int64_t const> const& selection, uint64_t offset = 0)
3580 : FilteredBase<T>(std::move(tables), selection, offset) {}
3581
3583 {
3584 Filtered<T> copy(*this);
3585 copy.sumWithSelection(selection);
3586 return copy;
3587 }
3588
3589 Filtered<T> operator+(gsl::span<int64_t const> const& selection)
3590 {
3591 Filtered<T> copy(*this);
3592 copy.sumWithSelection(selection);
3593 return copy;
3594 }
3595
3597 {
3598 return operator+(other.getSelectedRows());
3599 }
3600
3602 {
3603 this->sumWithSelection(selection);
3604 return *this;
3605 }
3606
3607 Filtered<T> operator+=(gsl::span<int64_t const> const& selection)
3608 {
3609 this->sumWithSelection(selection);
3610 return *this;
3611 }
3612
3614 {
3615 return operator+=(other.getSelectedRows());
3616 }
3617
3619 {
3620 Filtered<T> copy(*this);
3621 copy.intersectWithSelection(selection);
3622 return copy;
3623 }
3624
3625 Filtered<T> operator*(gsl::span<int64_t const> const& selection)
3626 {
3627 Filtered<T> copy(*this);
3628 copy.intersectWithSelection(selection);
3629 return copy;
3630 }
3631
3633 {
3634 return operator*(other.getSelectedRows());
3635 }
3636
3638 {
3639 this->intersectWithSelection(selection);
3640 return *this;
3641 }
3642
3643 Filtered<T> operator*=(gsl::span<int64_t const> const& selection)
3644 {
3645 this->intersectWithSelection(selection);
3646 return *this;
3647 }
3648
3650 {
3651 return operator*=(other.getSelectedRows());
3652 }
3653
3655 {
3656 auto it = unfiltered_iterator{this->cached_begin()};
3657 it.setCursor(i);
3658 return it;
3659 }
3660
3661 using FilteredBase<T>::getSelectedRows;
3662
3663 auto rawSlice(uint64_t start, uint64_t end) const
3664 {
3665 SelectionVector newSelection;
3666 newSelection.resize(static_cast<int64_t>(end - start + 1));
3667 std::iota(newSelection.begin(), newSelection.end(), start);
3668 return self_t{{this->asArrowTable()}, std::move(newSelection), 0};
3669 }
3670
3671 auto emptySlice() const
3672 {
3673 return self_t{{this->asArrowTable()}, SelectionVector{}, 0};
3674 }
3675
3676 template <typename T1>
3677 auto rawSliceBy(o2::framework::Preslice<T1> const& container, int value) const
3678 {
3679 return (table_t)this->sliceBy(container, value);
3680 }
3681
3683 {
3684 return doFilteredSliceByCached(this, node, value, cache);
3685 }
3686
3688 {
3689 return doSliceByCachedUnsorted(this, node, value, cache);
3690 }
3691
3692 template <typename T1, bool OPT>
3694 {
3695 return doFilteredSliceBy(this, container, value);
3696 }
3697
3698 template <typename T1, bool OPT>
3700 {
3701 return doSliceBy(this, container, value);
3702 }
3703
3705 {
3706 auto t = o2::soa::select(*this, f);
3707 copyIndexBindings(t);
3708 return t;
3709 }
3710};
3711
3712template <typename T>
3713class Filtered<Filtered<T>> : public FilteredBase<typename T::table_t>
3714{
3715 public:
3717 using base_t = T;
3719 using columns_t = typename T::columns_t;
3720
3721 using iterator = typename T::template iterator_template_o<FilteredIndexPolicy, self_t>;
3722 using unfiltered_iterator = typename T::template iterator_template_o<DefaultIndexPolicy, self_t>;
3724
3726 {
3727 return iterator(this->cached_begin());
3728 }
3729
3731 {
3732 return const_iterator(this->cached_begin());
3733 }
3734
3735 Filtered(std::vector<Filtered<T>>&& tables, gandiva::Selection const& selection, uint64_t offset = 0)
3736 : FilteredBase<typename T::table_t>(std::move(extractTablesFromFiltered(tables)), selection, offset)
3737 {
3738 for (auto& table : tables) {
3739 *this *= table;
3740 }
3741 }
3742
3743 Filtered(std::vector<Filtered<T>>&& tables, SelectionVector&& selection, uint64_t offset = 0)
3744 : FilteredBase<typename T::table_t>(std::move(extractTablesFromFiltered(tables)), std::forward<SelectionVector>(selection), offset)
3745 {
3746 for (auto& table : tables) {
3747 *this *= table;
3748 }
3749 }
3750
3751 Filtered(std::vector<Filtered<T>>&& tables, gsl::span<int64_t const> const& selection, uint64_t offset = 0)
3752 : FilteredBase<typename T::table_t>(std::move(extractTablesFromFiltered(tables)), selection, offset)
3753 {
3754 for (auto& table : tables) {
3755 *this *= table;
3756 }
3757 }
3758
3760 {
3761 Filtered<Filtered<T>> copy(*this);
3762 copy.sumWithSelection(selection);
3763 return copy;
3764 }
3765
3766 Filtered<Filtered<T>> operator+(gsl::span<int64_t const> const& selection)
3767 {
3768 Filtered<Filtered<T>> copy(*this);
3769 copy.sumWithSelection(selection);
3770 return copy;
3771 }
3772
3774 {
3775 return operator+(other.getSelectedRows());
3776 }
3777
3779 {
3780 this->sumWithSelection(selection);
3781 return *this;
3782 }
3783
3784 Filtered<Filtered<T>> operator+=(gsl::span<int64_t const> const& selection)
3785 {
3786 this->sumWithSelection(selection);
3787 return *this;
3788 }
3789
3791 {
3792 return operator+=(other.getSelectedRows());
3793 }
3794
3796 {
3797 Filtered<Filtered<T>> copy(*this);
3798 copy.intersectionWithSelection(selection);
3799 return copy;
3800 }
3801
3802 Filtered<Filtered<T>> operator*(gsl::span<int64_t const> const& selection)
3803 {
3804 Filtered<Filtered<T>> copy(*this);
3805 copy.intersectionWithSelection(selection);
3806 return copy;
3807 }
3808
3810 {
3811 return operator*(other.getSelectedRows());
3812 }
3813
3815 {
3816 this->intersectWithSelection(selection);
3817 return *this;
3818 }
3819
3820 Filtered<Filtered<T>> operator*=(gsl::span<int64_t const> const& selection)
3821 {
3822 this->intersectWithSelection(selection);
3823 return *this;
3824 }
3825
3827 {
3828 return operator*=(other.getSelectedRows());
3829 }
3830
3832 {
3833 auto it = unfiltered_iterator{this->cached_begin()};
3834 it.setCursor(i);
3835 return it;
3836 }
3837
3838 auto rawSlice(uint64_t start, uint64_t end) const
3839 {
3840 SelectionVector newSelection;
3841 newSelection.resize(static_cast<int64_t>(end - start + 1));
3842 std::iota(newSelection.begin(), newSelection.end(), start);
3843 return self_t{{this->asArrowTable()}, std::move(newSelection), 0};
3844 }
3845
3846 auto emptySlice() const
3847 {
3848 return self_t{{this->asArrowTable()}, SelectionVector{}, 0};
3849 }
3850
3852 {
3853 return doFilteredSliceByCached(this, node, value, cache);
3854 }
3855
3857 {
3858 return doSliceByCachedUnsorted(this, node, value, cache);
3859 }
3860
3861 template <typename T1, bool OPT>
3863 {
3864 return doFilteredSliceBy(this, container, value);
3865 }
3866
3867 template <typename T1, bool OPT>
3869 {
3870 return doSliceBy(this, container, value);
3871 }
3872
3873 private:
3874 std::vector<std::shared_ptr<arrow::Table>> extractTablesFromFiltered(std::vector<Filtered<T>>& tables)
3875 {
3876 std::vector<std::shared_ptr<arrow::Table>> outTables;
3877 for (auto& table : tables) {
3878 outTables.push_back(table.asArrowTable());
3879 }
3880 return outTables;
3881 }
3882};
3883
3889template <typename L, typename D, typename O, typename Key, typename H, typename... Ts>
3890struct IndexTable : Table<L, D, O> {
3891 using self_t = IndexTable<L, D, O, Key, H, Ts...>;
3896 using first_t = typename H::binding_t;
3897 using rest_t = framework::pack<typename Ts::binding_t...>;
3898
3899 IndexTable(std::shared_ptr<arrow::Table> table, uint64_t offset = 0)
3900 : base_t{table, offset}
3901 {
3902 }
3903
3904 IndexTable(std::vector<std::shared_ptr<arrow::Table>> tables, uint64_t offset = 0)
3905 : base_t{tables[0], offset}
3906 {
3907 }
3908
3909 IndexTable(IndexTable const&) = default;
3911 IndexTable& operator=(IndexTable const&) = default;
3913
3918};
3919
3920template <typename T, bool APPLY>
3921struct SmallGroupsBase : public Filtered<T> {
3922 static constexpr bool applyFilters = APPLY;
3923 SmallGroupsBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, gandiva::Selection const& selection, uint64_t offset = 0)
3924 : Filtered<T>(std::move(tables), selection, offset) {}
3925
3926 SmallGroupsBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, SelectionVector&& selection, uint64_t offset = 0)
3927 : Filtered<T>(std::move(tables), std::forward<SelectionVector>(selection), offset) {}
3928
3929 SmallGroupsBase(std::vector<std::shared_ptr<arrow::Table>>&& tables, gsl::span<int64_t const> const& selection, uint64_t offset = 0)
3930 : Filtered<T>(std::move(tables), selection, offset) {}
3931};
3932
3933template <typename T>
3935
3936template <typename T>
3938
3939template <typename T>
3940concept is_smallgroups = requires {
3941 []<typename B, bool A>(SmallGroupsBase<B, A>*) {}(std::declval<T*>());
3942};
3943} // namespace o2::soa
3944
3945#endif // O2_FRAMEWORK_ASOA_H_
#define O2HASH(_Str_)
Pre-declare Hash specialization for a generic string.
Definition ASoA.h:276
#define O2ORIGIN(_Str_)
Pre-declare Hash specialization for an origin string.
Definition ASoA.h:284
consteval auto getIndexTargets()
SLICE.
Definition ASoA.h:2418
o2::monitoring::tags::Key Key
#define O2_BUILTIN_UNREACHABLE
#define O2_BUILTIN_LIKELY(x)
#define O2_BUILTIN_UNLIKELY(x)
Hit operator+(const Hit &lhs, const Hit &rhs)
Definition Hit.cxx:46
int32_t i
bool o
uint16_t pos
Definition RawData.h:3
uint32_t res
Definition RawData.h:0
uint32_t c
Definition RawData.h:2
uint32_t version
Definition RawData.h:8
TBranch * ptr
void merge(Options const &options)
StringRef key
Definition A.h:16
Definition B.h:16
Class for time synchronization of RawReader instances.
int64_t const * mCurrentPos
Definition ASoA.h:592
ColumnIterator(arrow::ChunkedArray const *column)
Definition ASoA.h:493
ColumnIterator(ColumnIterator< T, ChunkingPolicy > const &)=default
void moveToEnd()
Move the iterator to the end of the column.
Definition ASoA.h:551
ColumnIterator< T > & moveToPos()
Definition ASoA.h:585
ColumnIterator< T, ChunkingPolicy > & operator=(ColumnIterator< T, ChunkingPolicy > const &)=default
unwrap_t< T > const * mCurrent
Definition ASoA.h:591
unwrap_t< T > const * mLast
Definition ASoA.h:593
ColumnIterator(ColumnIterator< T, ChunkingPolicy > &&)=default
void prevChunk() const
Definition ASoA.h:526
arrow::ChunkedArray const * mColumn
Definition ASoA.h:594
ColumnIterator< T, ChunkingPolicy > & operator=(ColumnIterator< T, ChunkingPolicy > &&)=default
void moveToChunk(int chunk)
Definition ASoA.h:537
void nextChunk() const
Move the iterator to the next chunk.
Definition ASoA.h:515
bool isCached() const
Definition ASoA.h:3524
auto sliceByCachedUnsorted(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3451
FilteredBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, gsl::span< int64_t const > const &selection, uint64_t offset=0)
Definition ASoA.h:3321
int64_t tableSize() const
Definition ASoA.h:3374
FilteredBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, gandiva::Selection const &selection, uint64_t offset=0)
Definition ASoA.h:3297
auto & cached_begin()
Definition ASoA.h:3354
typename T::external_index_columns_t external_index_columns_t
Definition ASoA.h:3291
auto select(framework::expressions::Filter const &f) const
Definition ASoA.h:3468
static auto getSpan(gandiva::Selection const &sel)
Definition ASoA.h:3397
int64_t size() const
Definition ASoA.h:3369
T::template iterator_template_o< FilteredIndexPolicy, self_t > iterator
Definition ASoA.h:3293
auto rawSliceBy(o2::framework::Preslice< T1 > const &container, int value) const
Definition ASoA.h:3441
T::template iterator_template_o< DefaultIndexPolicy, self_t > unfiltered_iterator
Definition ASoA.h:3294
void copyIndexBindings(T1 &dest) const
Definition ASoA.h:3435
auto const & getSelectedRows() const
Definition ASoA.h:3379
typename T::columns_t columns_t
Definition ASoA.h:3289
auto emptySlice() const
Definition ASoA.h:3392
void bindExternalIndices(TA *... current)
Definition ASoA.h:3411
void sumWithSelection(SelectionVector const &selection)
Definition ASoA.h:3484
void bindInternalIndicesTo(I const *ptr)
Definition ASoA.h:3423
void intersectWithSelection(SelectionVector const &selection)
Definition ASoA.h:3494
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicyGeneral, OPT > const &container, int value) const
Definition ASoA.h:3463
auto rawSlice(uint64_t start, uint64_t end) const
Definition ASoA.h:3384
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicySorted, OPT > const &container, int value) const
Definition ASoA.h:3457
iterator iteratorAt(uint64_t i) const
Definition ASoA.h:3364
iterator const_iterator
Definition ASoA.h:3295
typename T::table_t table_t
Definition ASoA.h:3287
typename T::persistent_columns_t persistent_columns_t
Definition ASoA.h:3290
RowViewSentinel end() const
Definition ASoA.h:3349
void intersectWithSelection(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3514
FilteredBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, SelectionVector &&selection, uint64_t offset=0)
Definition ASoA.h:3308
void bindExternalIndicesRaw(std::vector< o2::soa::Binding > &&ptrs)
Definition ASoA.h:3417
const_iterator begin() const
Definition ASoA.h:3337
unfiltered_iterator rawIteratorAt(uint64_t i) const
Definition ASoA.h:3342
int isInSelectedRows(int i) const
Definition ASoA.h:3475
auto sliceByCached(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3446
void doCopyIndexBindings(framework::pack< Cs... >, T1 &dest) const
Definition ASoA.h:3429
void sumWithSelection(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3504
iterator begin()
Definition ASoA.h:3332
auto const & cached_begin() const
Definition ASoA.h:3359
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicySorted, OPT > const &container, int value) const
Definition ASoA.h:3862
typename FilteredBase< typename T::table_t >::table_t table_t
Definition ASoA.h:3718
Filtered< Filtered< T > > operator+=(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3784
typename T::template iterator_template_o< DefaultIndexPolicy, self_t > unfiltered_iterator
Definition ASoA.h:3722
Filtered< Filtered< T > > operator+(SelectionVector const &selection)
Definition ASoA.h:3759
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicyGeneral, OPT > const &container, int value) const
Definition ASoA.h:3868
Filtered< Filtered< T > > operator*(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3802
typename T::template iterator_template_o< FilteredIndexPolicy, self_t > iterator
Definition ASoA.h:3721
typename T::columns_t columns_t
Definition ASoA.h:3719
const_iterator begin() const
Definition ASoA.h:3730
Filtered(std::vector< Filtered< T > > &&tables, SelectionVector &&selection, uint64_t offset=0)
Definition ASoA.h:3743
unfiltered_iterator rawIteratorAt(uint64_t i) const
Definition ASoA.h:3831
Filtered< Filtered< T > > operator+=(Filtered< T > const &other)
Definition ASoA.h:3790
Filtered< Filtered< T > > operator+=(SelectionVector const &selection)
Definition ASoA.h:3778
Filtered< Filtered< T > > operator*=(SelectionVector const &selection)
Definition ASoA.h:3814
auto rawSlice(uint64_t start, uint64_t end) const
Definition ASoA.h:3838
Filtered< Filtered< T > > operator+(Filtered< T > const &other)
Definition ASoA.h:3773
Filtered(std::vector< Filtered< T > > &&tables, gandiva::Selection const &selection, uint64_t offset=0)
Definition ASoA.h:3735
Filtered< Filtered< T > > operator*=(Filtered< T > const &other)
Definition ASoA.h:3826
auto sliceByCached(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3851
Filtered< Filtered< T > > operator*=(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3820
Filtered< Filtered< T > > operator*(Filtered< T > const &other)
Definition ASoA.h:3809
Filtered< Filtered< T > > operator+(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3766
Filtered< Filtered< T > > operator*(SelectionVector const &selection)
Definition ASoA.h:3795
auto sliceByCachedUnsorted(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3856
Filtered(std::vector< Filtered< T > > &&tables, gsl::span< int64_t const > const &selection, uint64_t offset=0)
Definition ASoA.h:3751
Filtered< T > operator+=(Filtered< T > const &other)
Definition ASoA.h:3613
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicyGeneral, OPT > const &container, int value) const
Definition ASoA.h:3699
Filtered(std::vector< std::shared_ptr< arrow::Table > > &&tables, SelectionVector &&selection, uint64_t offset=0)
Definition ASoA.h:3576
iterator const_iterator
Definition ASoA.h:3561
Filtered(std::vector< std::shared_ptr< arrow::Table > > &&tables, gsl::span< int64_t const > const &selection, uint64_t offset=0)
Definition ASoA.h:3579
iterator begin()
Definition ASoA.h:3563
auto sliceByCachedUnsorted(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3687
Filtered< T > operator+(Filtered< T > const &other)
Definition ASoA.h:3596
Filtered< T > operator*=(SelectionVector const &selection)
Definition ASoA.h:3637
auto emptySlice() const
Definition ASoA.h:3671
const_iterator begin() const
Definition ASoA.h:3568
T::template iterator_template_o< FilteredIndexPolicy, self_t > iterator
Definition ASoA.h:3559
auto sliceByCached(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3682
auto select(framework::expressions::Filter const &f) const
Definition ASoA.h:3704
Filtered< T > operator+(SelectionVector const &selection)
Definition ASoA.h:3582
Filtered< T > operator*(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3625
T::template iterator_template_o< DefaultIndexPolicy, self_t > unfiltered_iterator
Definition ASoA.h:3560
unfiltered_iterator rawIteratorAt(uint64_t i) const
Definition ASoA.h:3654
Filtered< T > operator*(SelectionVector const &selection)
Definition ASoA.h:3618
Filtered< T > operator*=(Filtered< T > const &other)
Definition ASoA.h:3649
Filtered< T > operator*(Filtered< T > const &other)
Definition ASoA.h:3632
auto rawSliceBy(o2::framework::Preslice< T1 > const &container, int value) const
Definition ASoA.h:3677
Filtered< T > operator+=(SelectionVector const &selection)
Definition ASoA.h:3601
Filtered< T > operator*=(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3643
Filtered< T > operator+(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3589
Filtered< T > operator+=(gsl::span< int64_t const > const &selection)
Definition ASoA.h:3607
auto rawSlice(uint64_t start, uint64_t end) const
Definition ASoA.h:3663
Filtered(std::vector< std::shared_ptr< arrow::Table > > &&tables, gandiva::Selection const &selection, uint64_t offset=0)
Definition ASoA.h:3573
auto sliceBy(o2::framework::PresliceBase< T1, framework::PreslicePolicySorted, OPT > const &container, int value) const
Definition ASoA.h:3693
typename T::table_t table_t
Definition ASoA.h:3556
typename T::columns_t columns_t
Definition ASoA.h:3557
decltype([]< typename... C >(framework::pack< C... > &&) -> framework::selected_pack< soa::is_self_index_t, C... > {}(columns_t{})) internal_index_columns_t
Definition ASoA.h:1720
void bindInternalIndicesExplicit(o2::soa::Binding binding)
Definition ASoA.h:2047
auto & cached_begin()
Definition ASoA.h:1961
iterator iteratorAt(uint64_t i) const
Definition ASoA.h:1990
decltype([]< typename... C >(framework::pack< C... > &&) -> framework::selected_pack< soa::is_persistent_column_t, C... > {}(columns_t{})) persistent_columns_t
Definition ASoA.h:1716
static constexpr const auto ref
Definition ASoA.h:1678
auto offset() const
Return offset.
Definition ASoA.h:2018
static consteval bool hasOriginal()
Definition ASoA.h:1709
unfiltered_iterator begin()
Definition ASoA.h:1971
int64_t tableSize() const
Definition ASoA.h:2028
auto rawSlice(uint64_t start, uint64_t end) const
Definition ASoA.h:2098
void bindExternalIndices(TA *... current)
Definition ASoA.h:2036
auto select(framework::expressions::Filter const &f) const
Definition ASoA.h:2075
unfiltered_iterator unfiltered_const_iterator
Definition ASoA.h:1902
static consteval auto full_iter()
Definition ASoA.h:1881
auto const & cached_begin() const
Definition ASoA.h:1966
auto emptySlice() const
Definition ASoA.h:2103
auto sliceByCachedUnsorted(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:2087
void bindExternalIndicesRaw(std::vector< o2::soa::Binding > &&ptrs)
Definition ASoA.h:2058
auto sliceBy(o2::framework::PresliceBase< T1, Policy, OPT > const &container, int value) const
Definition ASoA.h:2093
static constexpr auto hashes()
Definition ASoA.h:1904
iterator unfiltered_iterator
Definition ASoA.h:1900
std::shared_ptr< arrow::Table > asArrowTable() const
Return a type erased arrow table backing store for / the type safe table.
Definition ASoA.h:2013
void doCopyIndexBindings(framework::pack< Cs... >, T &dest) const
Definition ASoA.h:2064
int64_t size() const
Size of the table, in rows.
Definition ASoA.h:2023
Table(std::vector< std::shared_ptr< arrow::Table > > &&tables, uint64_t offset=0)
Definition ASoA.h:1935
decltype(getColumns< ref, Ts... >()) columns_t
Definition ASoA.h:1714
arrow::ChunkedArray * getIndexToKey()
Definition ASoA.h:1942
RowViewSentinel end()
Definition ASoA.h:1976
decltype([]< typename... C >(framework::pack< C... >) -> framework::pack< typename C::type... > {}(persistent_columns_t{})) column_types
Definition ASoA.h:1717
static consteval auto isIndexTargetOf()
Definition ASoA.h:1686
iterator_template_o< FilteredIndexPolicy, table_t > filtered_iterator
Definition ASoA.h:1898
unfiltered_const_iterator begin() const
Definition ASoA.h:2002
void copyIndexBindings(T &dest) const
Definition ASoA.h:2070
Table(std::vector< std::shared_ptr< arrow::Table > > &&tables, uint64_t offset=0)
Definition ASoA.h:1929
Table< L, D, O, Ts... > self_t
Definition ASoA.h:1679
static consteval auto isIndexTargetOf()
Definition ASoA.h:1693
decltype(full_iter< IP, Parent >()) iterator_template_o
Definition ASoA.h:1895
void doBindInternalIndicesExplicit(framework::pack< Cs... >, o2::soa::Binding binding)
Definition ASoA.h:2053
static constexpr const auto originals
Definition ASoA.h:1682
decltype([]< typename... C >(framework::pack< C... > &&) -> framework::selected_pack< soa::is_external_index_t, C... > {}(columns_t{})) external_index_columns_t
Definition ASoA.h:1719
Table(std::shared_ptr< arrow::Table > table, uint64_t offset=0)
Definition ASoA.h:1909
RowViewSentinel end() const
Definition ASoA.h:2007
auto sliceByCached(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:2082
void bindInternalIndicesTo(I const *ptr)
Definition ASoA.h:2042
unfiltered_iterator rawIteratorAt(uint64_t i) const
Definition ASoA.h:1995
filtered_iterator filtered_begin(gsl::span< int64_t const > selection)
Definition ASoA.h:1981
iterator_template_o< DefaultIndexPolicy, table_t > iterator
Definition ASoA.h:1897
hash identification concepts
Definition ASoA.h:363
Helper to check if a type T is an iterator.
Definition ASoA.h:1251
column identification concepts
Definition ASoA.h:185
GLint GLenum GLint x
Definition glcorearb.h:403
GLenum func
Definition glcorearb.h:778
GLint GLsizei count
Definition glcorearb.h:399
GLsizeiptr size
Definition glcorearb.h:659
GLuint GLsizei const GLuint const GLintptr * offsets
Definition glcorearb.h:2595
GLuint GLuint end
Definition glcorearb.h:469
GLenum array
Definition glcorearb.h:4274
GLuint index
Definition glcorearb.h:781
GLdouble f
Definition glcorearb.h:310
GLboolean GLboolean GLboolean b
Definition glcorearb.h:1233
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLenum target
Definition glcorearb.h:1641
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
GLboolean * data
Definition glcorearb.h:298
GLintptr offset
Definition glcorearb.h:660
GLuint GLsizei GLsizei * length
Definition glcorearb.h:790
GLuint GLsizei const GLchar * label
Definition glcorearb.h:2519
GLsizei GLenum const void * indices
Definition glcorearb.h:400
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLboolean r
Definition glcorearb.h:1233
GLuint start
Definition glcorearb.h:469
GLenum GLenum GLsizei len
Definition glcorearb.h:4232
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
GLint ref
Definition glcorearb.h:291
std::shared_ptr< gandiva::SelectionVector > Selection
Definition Expressions.h:46
consteval const char * origin_str()
Definition ASoA.h:344
consteval const char * signature()
Definition ASoA.h:356
consteval auto filterForKey()
Filter TableRef array for compatibility with Key table.
Definition ASoA.h:264
consteval const char * label()
Definition ASoA.h:338
consteval header::DataOrigin origin()
Definition ASoA.h:350
std::shared_ptr< arrow::DataType > concreteArrowType(atype::type type)
gandiva::Selection createSelection(std::shared_ptr< arrow::Table > const &table, Filter const &expression)
Function for creating gandiva selection from our internal filter tree.
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
std::decay_t< decltype(select_pack< Condition >(pack<>{}, Pack{}, CondPack{}))> selected_pack_multicondition
Definition Pack.h:181
std::decay_t< decltype(prune_voids_pack(pack<>{}, with_condition_pack< Condition, Types... >{}))> selected_pack
Definition Pack.h:179
decltype(intersected_pack(Ps{}...)) full_intersected_pack_t
Definition Pack.h:283
typename pack_element< I, T >::type pack_element_t
Definition Pack.h:56
consteval size_t has_type_at_v(pack< Ts... >)
Definition Pack.h:228
constexpr std::size_t pack_size(pack< Ts... > const &)
template function to determine number of types in a pack
Definition Pack.h:28
decltype(concatenate_pack_unique(Ps{}...)) concatenated_pack_unique_t
Definition Pack.h:319
std::pair< std::string, std::string > StringPair
std::string strToUpper(std::string &&str)
Definition ASoA.cxx:189
typename pack_element< 0, T >::type pack_head_t
Definition Pack.h:59
std::string cutString(std::string &&str)
Definition ASoA.cxx:180
std::vector< std::vector< int64_t > > ListVector
Descriptor< gSizeDataDescriptionString > DataDescription
Definition DataHeader.h:551
std::array< std::shared_ptr< arrow::Array >, sizeof...(Cs)> getChunks(arrow::Table *table, framework::pack< Cs... >, uint64_t ci)
Definition ASoA.h:2142
ColumnGetterFunction< R, typename T::iterator > getColumnGetterByLabel(const std::string_view &targetColumnLabel)
Definition ASoA.h:2250
std::array< arrow::ChunkedArray *, sizeof...(Cs)> getArrowColumns(arrow::Table *table, framework::pack< Cs... >)
Definition ASoA.h:2136
std::tuple< typename Cs::type... > getRowData(arrow::Table *table, T rowIterator, uint64_t ci=std::numeric_limits< uint64_t >::max(), uint64_t ai=std::numeric_limits< uint64_t >::max(), uint64_t globalIndex=std::numeric_limits< uint64_t >::max())
Definition ASoA.h:2177
C::type getSingleRowData(arrow::Table *table, T &rowIterator, uint64_t ci=std::numeric_limits< uint64_t >::max(), uint64_t ai=std::numeric_limits< uint64_t >::max(), uint64_t globalIndex=std::numeric_limits< uint64_t >::max())
Definition ASoA.h:2148
consteval auto computeOriginals()
Definition ASoA.h:1660
auto createFieldsFromColumns(framework::pack< C... >)
Definition ASoA.h:406
SelectionVector selectionToVector(gandiva::Selection const &sel)
Definition ASoA.cxx:43
constexpr bool is_persistent_v
Definition ASoA.h:188
constexpr bool is_ng_index_equivalent_v
Definition ASoA.h:445
consteval auto remove_if(L l)
Definition ASoA.h:126
constexpr auto join(Ts const &... t)
Definition ASoA.h:3233
auto doSliceBy(T const *table, o2::framework::PresliceBase< C, Policy, OPT > const &container, int value)
Definition ASoA.h:1504
SelectionVector sliceSelection(gsl::span< int64_t const > const &mSelectedRows, int64_t nrows, uint64_t offset)
Definition ASoA.cxx:53
void notBoundTable(const char *tableName)
Definition ASoA.cxx:161
auto doFilteredSliceBy(T const *table, o2::framework::PresliceBase< C, framework::PreslicePolicySorted, OPT > const &container, int value)
Definition ASoA.h:1570
constexpr auto concat(Ts const &... t)
Definition ASoA.h:3277
consteval auto intersectOriginals()
Definition ASoA.h:161
consteval auto getColumns()
Definition ASoA.h:1632
std::vector< int64_t > SelectionVector
Definition ASoA.h:411
std::conditional_t< is_binding_compatible_v< T, typename B::binding_t >(), std::true_type, std::false_type > is_binding_compatible
Definition ASoA.h:1286
consteval auto mergeOriginals()
Definition ASoA.h:144
constexpr bool is_soa_filtered_v
Definition ASoA.h:1485
typename std::conditional_t< is_index_column< C >, std::true_type, std::false_type > is_external_index_t
Definition ASoA.h:203
void missingFilterDeclaration(int hash, int ai)
Definition ASoA.cxx:28
auto doSliceByCachedUnsorted(T const *table, framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache)
Definition ASoA.h:1602
auto doSliceByCached(T const *table, framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache)
Definition ASoA.h:1583
void accessingInvalidIndexFor(const char *getter)
Definition ASoA.cxx:20
auto select(T const &t, framework::expressions::Filter const &f)
Definition ASoA.h:1618
consteval auto base_iter(framework::pack< C... > &&) -> TableIterator< D, O, IP, C... >
Definition ASoA.h:1626
constexpr bool is_index_equivalent_v
Definition ASoA.h:442
constexpr bool is_soa_join_v
Definition ASoA.h:3242
void dereferenceWithWrongType(const char *getter, const char *target)
Definition ASoA.cxx:24
std::conditional_t< is_indexing_column< T >, std::true_type, std::false_type > is_indexing_t
Definition ASoA.h:821
auto doSliceByHelper(T const *table, gsl::span< const int64_t > const &selection)
Definition ASoA.h:1520
consteval bool is_binding_compatible_v()
Definition ASoA.h:1280
auto prepareFilteredSlice(T const *table, std::shared_ptr< arrow::Table > slice, uint64_t offset)
Definition ASoA.h:1555
void emptyColumnLabel()
Definition ASoA.cxx:38
typename unwrap< T >::type unwrap_t
Definition ASoA.h:477
void getterNotFound(const char *targetColumnLabel)
Definition ASoA.cxx:33
auto doFilteredSliceByCached(T const *table, framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache)
Definition ASoA.h:1593
void missingOptionalPreslice(const char *label, const char *key)
Definition ASoA.cxx:171
consteval bool is_compatible()
Definition ASoA.h:1267
arrow::ChunkedArray * getIndexFromLabel(arrow::Table *table, std::string_view label)
Definition ASoA.cxx:140
consteval auto merge()
Helpers to manipulate TableRef arrays.
Definition ASoA.h:103
consteval auto merge_if(L l)
Definition ASoA.h:114
std::conditional_t< is_persistent_column< C >, std::true_type, std::false_type > is_persistent_column_t
Definition ASoA.h:191
typename arrow_array_for< T >::type arrow_array_for_t
Definition ArrowTypes.h:112
void notFoundColumn(const char *label, const char *key)
Definition ASoA.cxx:166
std::conditional_t< is_dynamic_column< T >, std::true_type, std::false_type > is_dynamic_t
Definition ASoA.h:815
typename std::conditional_t< is_self_index_column< C >, std::true_type, std::false_type > is_self_index_t
Definition ASoA.h:206
Defining DataPointCompositeObject explicitly as copiable.
FIXME: do not use data model tables.
Definition list.h:40
static constexpr uint32_t hash
Definition ASoA.h:258
static constexpr char const *const str
Definition ASoA.h:259
Base type for table metadata.
Definition ASoA.h:213
static consteval int getIndexPosToKey()
Definition ASoA.h:232
static consteval int getIndexPosToKey_impl()
Definition ASoA.h:238
framework::selected_pack< soa::is_self_index_t, Cs... > internal_index_columns_t
Definition ASoA.h:217
framework::selected_pack< soa::is_persistent_column_t, Cs... > persistent_columns_t
Definition ASoA.h:215
framework::selected_pack< soa::is_external_index_t, Cs... > external_index_columns_t
Definition ASoA.h:216
static consteval std::array< bool, sizeof...(PCs)> getMap(framework::pack< PCs... >)
Definition ASoA.h:220
SliceInfoPtr getCacheFor(StringPair const &bindingKey) const
SliceInfoUnsortedPtr getCacheUnsortedFor(StringPair const &bindingKey) const
std::shared_ptr< arrow::Table > getSliceFor(int value, std::shared_ptr< arrow::Table > const &input, uint64_t &offset) const
Definition ASoA.h:1432
static constexpr bool optional
Definition ASoA.h:1422
PresliceBase(expressions::BindingNode index_)
Definition ASoA.h:1427
gsl::span< const int64_t > getSliceFor(int value) const
Definition ASoA.h:1442
const std::string binding
Definition ASoA.h:1425
StringPair const & getBindingKey() const
Definition ASoA.cxx:200
const std::string binding
Definition ASoA.h:1399
SliceInfoUnsortedPtr sliceInfo
Definition ASoA.h:1416
gsl::span< const int64_t > getSliceFor(int value) const
Definition ASoA.cxx:223
void updateSliceInfo(SliceInfoUnsortedPtr &&si)
Definition ASoA.cxx:210
void updateSliceInfo(SliceInfoPtr &&si)
Definition ASoA.cxx:205
std::shared_ptr< arrow::Table > getSliceFor(int value, std::shared_ptr< arrow::Table > const &input, uint64_t &offset) const
Definition ASoA.cxx:215
ArrowTableSlicingCache * ptr
Definition SliceCache.h:22
std::pair< int64_t, int64_t > getSliceFor(int value) const
An expression tree node corresponding to a column binding.
A struct, containing the root of the expression tree.
From https://en.cppreference.com/w/cpp/utility/variant/visit.
static std::shared_ptr< arrow::Table > joinTables(std::vector< std::shared_ptr< arrow::Table > > &&tables)
Definition ASoA.cxx:67
static std::shared_ptr< arrow::Table > concatTables(std::vector< std::shared_ptr< arrow::Table > > &&tables)
Definition ASoA.cxx:100
Type-checking index column binding.
Definition ASoA.h:382
std::span< TableRef const > refs
Definition ASoA.h:385
size_t hash
Definition ASoA.h:384
void const * ptr
Definition ASoA.h:383
void bind(T const *table)
Definition ASoA.h:388
T const * get() const
Definition ASoA.h:396
static constexpr bool chunked
Definition ASoA.h:451
arrow::ChunkedArray * second
Definition ASoA.h:1028
Column(ColumnIterator< T > const &it)
Definition ASoA.h:653
Column()=default
static constexpr const char *const & columnLabel()
Definition ASoA.h:666
ColumnIterator< T > const & getIterator() const
Definition ASoA.h:667
Column & operator=(Column const &)=default
INHERIT inherited_t
Definition ASoA.h:652
Column(Column &&)=default
static auto asArrowField()
Definition ASoA.h:672
Column & operator=(Column &&)=default
Column(Column const &)=default
ColumnIterator< T > mColumnIterator
Definition ASoA.h:679
table_t::template iterator_template< DefaultIndexPolicy, self_t, Ts... > iterator
Definition ASoA.h:3268
Concat(Ts const &... t, uint64_t offset=0)
Definition ASoA.h:3253
typename table_t::persistent_columns_t persistent_columns_t
Definition ASoA.h:3266
typename table_t::columns_t columns_t
Definition ASoA.h:3265
iterator const_iterator
Definition ASoA.h:3269
const_iterator unfiltered_const_iterator
Definition ASoA.h:3271
iterator unfiltered_iterator
Definition ASoA.h:3270
table_t::template iterator_template< FilteredIndexPolicy, self_t, Ts... > filtered_iterator
Definition ASoA.h:3272
Concat(std::vector< std::shared_ptr< arrow::Table > > &&tables, uint64_t offset=0)
Definition ASoA.h:3248
filtered_iterator filtered_const_iterator
Definition ASoA.h:3273
DefaultIndexPolicy(int64_t nRows, uint64_t offset)
Definition ASoA.h:950
friend bool operator==(DefaultIndexPolicy const &lh, DefaultIndexPolicy const &rh)
Definition ASoA.h:996
bool operator==(RowViewSentinel const &sentinel) const
Definition ASoA.h:1001
DefaultIndexPolicy(FilteredIndexPolicy const &other)
Definition ASoA.h:956
std::tuple< uint64_t const * > getOffsets() const
Definition ASoA.h:977
DefaultIndexPolicy & operator=(DefaultIndexPolicy &&)=default
void setCursor(int64_t i)
Definition ASoA.h:982
void limitRange(int64_t start, int64_t end)
Definition ASoA.h:962
DefaultIndexPolicy(DefaultIndexPolicy &&)=default
DefaultIndexPolicy(DefaultIndexPolicy const &)=default
DefaultIndexPolicy & operator=(DefaultIndexPolicy const &)=default
std::tuple< int64_t const *, int64_t const * > getIndices() const
Definition ASoA.h:971
DefaultIndexPolicy()=default
Needed to be able to copy the policy.
void moveByIndex(int64_t i)
Definition ASoA.h:986
static constexpr const char *const & columnLabel()
Definition ASoA.h:688
INHERIT inherited_t
Definition ASoA.h:686
void resetSelection(gsl::span< int64_t const > selection)
Definition ASoA.h:849
FilteredIndexPolicy & operator=(FilteredIndexPolicy &&)=default
std::tuple< int64_t const *, int64_t const * > getIndices() const
Definition ASoA.h:863
FilteredIndexPolicy & operator=(FilteredIndexPolicy const &)=default
auto getSelectionRow() const
Definition ASoA.h:913
friend bool operator==(FilteredIndexPolicy const &lh, FilteredIndexPolicy const &rh)
Definition ASoA.h:894
void setCursor(int64_t i)
Definition ASoA.h:882
FilteredIndexPolicy(FilteredIndexPolicy const &)=default
auto raw_size() const
Definition ASoA.h:923
bool operator==(RowViewSentinel const &sentinel) const
Definition ASoA.h:899
std::tuple< uint64_t const * > getOffsets() const
Definition ASoA.h:869
void limitRange(int64_t start, int64_t end)
Definition ASoA.h:874
FilteredIndexPolicy(gsl::span< int64_t const > selection, int64_t rows, uint64_t offset=0)
Definition ASoA.h:840
FilteredIndexPolicy(FilteredIndexPolicy &&)=default
void moveByIndex(int64_t i)
Definition ASoA.h:888
static constexpr bool chunked
Definition ASoA.h:457
static constexpr const char *const & columnLabel()
Definition ASoA.h:695
INHERIT inherited_t
Definition ASoA.h:693
uint64_t mOffset
Offset within a larger table.
Definition ASoA.h:827
int64_t mRowIndex
Position inside the current table.
Definition ASoA.h:825
IndexTable(std::vector< std::shared_ptr< arrow::Table > > tables, uint64_t offset=0)
Definition ASoA.h:3904
IndexTable(std::shared_ptr< arrow::Table > table, uint64_t offset=0)
Definition ASoA.h:3899
typename base_t::template iterator_template_o< DefaultIndexPolicy, self_t > iterator
Definition ASoA.h:3914
filtered_iterator const_filtered_iterator
Definition ASoA.h:3917
IndexTable(IndexTable &&)=default
IndexTable & operator=(IndexTable const &)=default
iterator const_iterator
Definition ASoA.h:3915
IndexTable & operator=(IndexTable &&)=default
typename H::binding_t first_t
Definition ASoA.h:3896
typename base_t::template iterator_template_o< FilteredIndexPolicy, self_t > filtered_iterator
Definition ASoA.h:3916
IndexTable(IndexTable const &)=default
int64_t type
Definition ASoA.h:792
Index(Index const &)=default
void setIndices(std::tuple< int64_t const *, int64_t const * > indices)
Definition ASoA.h:781
Index & operator=(Index &&)=default
int64_t index() const
Definition ASoA.h:754
Index()=default
Index(arrow::ChunkedArray const *)
Definition ASoA.h:740
int64_t filteredIndex() const
Definition ASoA.h:759
constexpr int64_t rangeEnd()
Definition ASoA.h:749
constexpr int64_t rangeStart()
Definition ASoA.h:744
Index(Index &&)=default
Index & operator=(Index const &)=default
int64_t index() const
Definition ASoA.h:770
int64_t globalIndex() const
Definition ASoA.h:764
std::tuple< int64_t const *, int64_t const * > rowIndices
Definition ASoA.h:794
int64_t offsets() const
Definition ASoA.h:776
void setOffsets(std::tuple< uint64_t const * > offsets)
Definition ASoA.h:786
static constexpr const char * mLabel
Definition ASoA.h:791
std::tuple< uint64_t const * > rowOffsets
Definition ASoA.h:797
iterator begin()
Definition ASoA.h:3174
iterator unfiltered_iterator
Definition ASoA.h:3169
auto sliceBy(o2::framework::PresliceBase< T1, Policy, OPT > const &container, int value) const
Definition ASoA.h:3195
auto rawSlice(uint64_t start, uint64_t end) const
Definition ASoA.h:3212
auto sliceByCachedUnsorted(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3189
JoinFull(std::vector< std::shared_ptr< arrow::Table > > &&tables, uint64_t offset=0)
Definition ASoA.h:3154
const_iterator unfiltered_const_iterator
Definition ASoA.h:3170
filtered_iterator filtered_const_iterator
Definition ASoA.h:3172
static consteval bool contains()
Definition ASoA.h:3223
iterator iteratorAt(uint64_t i) const
Definition ASoA.h:3207
typename table_t::columns_t columns_t
Definition ASoA.h:3165
table_t::template iterator_template< DefaultIndexPolicy, self_t, Ts... > iterator
Definition ASoA.h:3167
table_t::template iterator_template< FilteredIndexPolicy, self_t, Ts... > filtered_iterator
Definition ASoA.h:3171
static constexpr const auto originals
Definition ASoA.h:3164
JoinFull< D, Ts... > self_t
Definition ASoA.h:3162
const_iterator begin() const
Definition ASoA.h:3179
typename table_t::persistent_columns_t persistent_columns_t
Definition ASoA.h:3166
iterator const_iterator
Definition ASoA.h:3168
iterator rawIteratorAt(uint64_t i) const
Definition ASoA.h:3200
Table< o2::aod::Hash<"JOIN"_h >, D, o2::aod::Hash<"JOIN"_h >, Ts... > base
Definition ASoA.h:3147
JoinFull(std::shared_ptr< arrow::Table > &&table, uint64_t offset=0)
Definition ASoA.h:3149
auto emptySlice() const
Definition ASoA.h:3217
auto sliceByCached(framework::expressions::BindingNode const &node, int value, o2::framework::SliceCache &cache) const
Definition ASoA.h:3184
static constexpr const char *const & columnLabel()
Definition ASoA.h:702
INHERIT inherited_t
Definition ASoA.h:700
constexpr auto mark()
Definition ASoA.h:719
Marker & operator=(Marker const &)=default
size_t type
Definition ASoA.h:707
static constexpr auto value
Definition ASoA.h:709
static constexpr const char * mLabel
Definition ASoA.h:724
Marker(arrow::ChunkedArray const *)
Definition ASoA.h:718
Marker()=default
Marker(Marker const &)=default
Marker & operator=(Marker &&)=default
Marker(Marker &&)=default
int64_t const index
Definition ASoA.h:831
SmallGroupsBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, SelectionVector &&selection, uint64_t offset=0)
Definition ASoA.h:3926
SmallGroupsBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, gsl::span< int64_t const > const &selection, uint64_t offset=0)
Definition ASoA.h:3929
SmallGroupsBase(std::vector< std::shared_ptr< arrow::Table > > &&tables, gandiva::Selection const &selection, uint64_t offset=0)
Definition ASoA.h:3923
framework::selected_pack< soa::is_persistent_column_t, C... > persistent_columns_t
Definition ASoA.h:1045
void doSetCurrentIndexRaw(framework::pack< Cs... > p, std::vector< o2::soa::Binding > &&ptrs)
Definition ASoA.h:1168
void bindInternalIndices(I const *table)
Definition ASoA.h:1187
TableIterator(TableIterator< D, O, FilteredIndexPolicy, C... > const &other)
Definition ASoA.h:1086
TableIterator(self_t const &other)
Definition ASoA.h:1071
TableIterator operator-(int64_t dec) const
Definition ASoA.h:1128
TableIterator(arrow::ChunkedArray *columnData[sizeof...(C)], IP &&policy)
Definition ASoA.h:1057
TableIterator & operator=(TableIterator other)
Definition ASoA.h:1078
TableIterator & operator++()
Definition ASoA.h:1094
void doSetCurrentInternal(framework::pack< Cs... >, I const *ptr)
Definition ASoA.h:1174
auto getIndexBindingsImpl(framework::pack< Cs... >) const
Definition ASoA.h:1151
TableIterator operator--(int)
Definition ASoA.h:1113
void bindExternalIndicesRaw(std::vector< o2::soa::Binding > &&ptrs)
Definition ASoA.h:1181
TableIterator(arrow::ChunkedArray *columnData[sizeof...(C)], IP &&policy)
Definition ASoA.h:1050
decltype([]< typename... Cs >(framework::pack< Cs... >) -> framework::pack< typename Cs::binding_t... > {}(external_index_columns_t{})) bindings_pack_t
Definition ASoA.h:1048
void bindExternalIndices(TA *... current)
Definition ASoA.h:1162
auto getCurrent() const
Definition ASoA.h:1145
TableIterator & operator--()
Definition ASoA.h:1107
framework::selected_pack< soa::is_external_index_t, C... > external_index_columns_t
Definition ASoA.h:1046
TableIterator const & operator*() const
Definition ASoA.h:1133
TableIterator operator++(int)
Definition ASoA.h:1100
framework::selected_pack< soa::is_self_index_t, C... > internal_index_columns_t
Definition ASoA.h:1047
auto getIndexBindings() const
Definition ASoA.h:1156
void doSetCurrentIndex(framework::pack< CL... >, TA *current)
Definition ASoA.h:1139
TableIterator operator+(int64_t inc) const
Allow incrementing by more than one the iterator.
Definition ASoA.h:1121
Generic identifier for a table type.
Definition ASoA.h:57
constexpr TableRef & operator=(TableRef const &)=default
consteval TableRef()
Definition ASoA.h:58
uint32_t label_hash
Definition ASoA.h:72
constexpr bool descriptionCompatible(uint32_t _desc_hash) const noexcept
Definition ASoA.h:90
constexpr bool operator==(TableRef const &other) const noexcept
Definition ASoA.h:77
constexpr TableRef(TableRef &&)=default
uint32_t version
Definition ASoA.h:75
constexpr TableRef(TableRef const &)=default
consteval TableRef(uint32_t _label, uint32_t _desc, uint32_t _origin, uint32_t _version)
Definition ASoA.h:65
constexpr bool descriptionCompatible(TableRef const &other) const noexcept
Definition ASoA.h:85
uint32_t desc_hash
Definition ASoA.h:73
uint32_t origin_hash
Definition ASoA.h:74
constexpr TableRef & operator=(TableRef &&)=default
TableIteratorBase const & operator*() const
Definition ASoA.h:1871
TableIteratorBase(TableIteratorBase< IP, P, T... > const &other)
Definition ASoA.h:1779
typename Parent::columns_t columns_t
Definition ASoA.h:1726
TableIteratorBase(TableIteratorBase< IP, P, O1, Os... > const &other)
Definition ASoA.h:1765
typename Parent::external_index_columns_t external_index_columns_t
Definition ASoA.h:1727
TableIteratorBase operator-(int64_t dec) const
Definition ASoA.h:1866
void matchTo(TableIteratorBase< IP, P, T... > const &other)
Definition ASoA.h:1803
void matchTo(TableIteratorBase< IP, P, Os... > const &other)
Definition ASoA.h:1809
TableIteratorBase(TableIteratorBase< FilteredIndexPolicy, P, T... > other)
Definition ASoA.h:1791
std::array< B, sizeof...(CCs)> getValues() const
Definition ASoA.h:1848
static constexpr auto originals
Definition ASoA.h:1730
TableIteratorBase(TableIteratorBase< IP, P, T... > &&other) noexcept
Definition ASoA.h:1785
decltype([]< typename... C >(framework::pack< C... >) -> framework::pack< typename C::binding_t... > {}(external_index_columns_t{})) bindings_pack_t
Definition ASoA.h:1728
TableIteratorBase & operator=(TableIteratorBase< IP, P, Os... > other)
Definition ASoA.h:1742
TableIteratorBase(TableIteratorBase< IP, P, O1, Os... > &&other) noexcept
Definition ASoA.h:1772
TableIteratorBase & operator=(RowViewSentinel const &other)
Definition ASoA.h:1797
TableIteratorBase(arrow::ChunkedArray *columnData[framework::pack_size(columns_t{})], IP &&policy)
Definition ASoA.h:1736
TableIteratorBase & operator=(TableIteratorBase< IP, P, T... > other)
Definition ASoA.h:1750
TableIteratorBase operator+(int64_t inc) const
Allow incrementing by more than one the iterator.
Definition ASoA.h:1859
TableIteratorBase & operator=(TableIteratorBase< FilteredIndexPolicy, P, T... > other)
Definition ASoA.h:1757
unwrapper
Definition ASoA.h:462
VectorOfTObjectPtrs other
std::vector< ReadoutWindowData > rows
const std::string str