mirror of
https://github.com/matrix-construct/construct
synced 2024-11-26 08:42:34 +01:00
192 lines
3.2 KiB
C++
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;
|
|
}
|