0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-11-26 08:42:34 +01:00
construct/matrix/event_index.cc
2020-11-27 17:27:55 -08:00

192 lines
3.2 KiB
C++

// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.
ircd::m::event::idx
ircd::m::index(const event &event)
try
{
return index(event.event_id);
}
catch(const json::not_found &)
{
throw m::NOT_FOUND
{
"Cannot find index for event without an event_id."
};
}
ircd::m::event::idx
ircd::m::index(std::nothrow_t,
const event &event)
{
return index(std::nothrow, event.event_id);
}
ircd::m::event::idx
ircd::m::index(const event::id &event_id)
{
assert(event_id);
const auto ret
{
index(std::nothrow, event_id)
};
if(!ret)
throw m::NOT_FOUND
{
"no index found for %s",
string_view{event_id}
};
return ret;
}
ircd::m::event::idx
ircd::m::index(std::nothrow_t,
const event::id &event_id)
{
static auto &column
{
dbs::event_idx
};
bool found {false};
event::idx ret {0};
if(likely(event_id))
{
const mutable_buffer buf
{
reinterpret_cast<char *>(&ret), sizeof(ret)
};
const string_view val
{
read(column, event_id, found, buf)
};
}
return ret & boolmask<event::idx>(found);
}
bool
ircd::m::index(std::nothrow_t,
const event::id &event_id,
const event::closure_idx &closure)
{
auto &column
{
dbs::event_idx
};
return event_id && column(event_id, std::nothrow, [&closure]
(const string_view &value)
{
const event::idx event_idx
{
byte_view<event::idx>(value)
};
closure(event_idx);
});
}
size_t
ircd::m::index(const vector_view<event::idx> &out,
const event::prev &prev)
{
const auto num
{
std::min(prev.prev_events_count(), event::prev::MAX)
};
size_t i(0);
event::id event_id[num];
m::for_each(prev, [&num, &i, &event_id]
(const auto &prev_id)
{
assert(i < num);
event_id[i++] = prev_id;
return i < num;
});
assert(i == num);
return index(out, vector_view<const event::id>(event_id, i));
}
size_t
ircd::m::index(const vector_view<event::idx> &out_,
const vector_view<const event::id> &in_)
{
static const size_t batch_max
{
64
};
const size_t max
{
std::min(out_.size(), in_.size())
};
auto &column
{
dbs::event_idx
};
size_t ret(0);
for(size_t i(0); i < max; i += batch_max)
{
const size_t num
{
i + batch_max > max? max - i: batch_max
};
const vector_view<event::idx> out
(
out_.data() + i, num
);
const vector_view<const event::id> in
(
in_.data() + i, num
);
const vector_view<const string_view> keys
(
static_cast<const string_view *>(in.data()), in.size()
);
mutable_buffer buf[num];
for(size_t j(0); j < num; ++j)
{
buf[j] = mutable_buffer
{
reinterpret_cast<char *>(out.data() + j),
sizeof(out[j])
};
zero(buf[j]);
}
const vector_view<mutable_buffer> bufs
(
buf, num
);
const uint64_t found_mask
{
db::read(column, keys, bufs)
};
ret += __builtin_popcountl(found_mask);
}
return ret;
}