mirror of
https://github.com/MaskRay/ccls.git
synced 2025-01-31 09:50:26 +00:00
sketch partial ipc payload
This commit is contained in:
parent
0b44f72ed8
commit
c079ab45b3
107
ipc.cc
107
ipc.cc
@ -2,6 +2,45 @@
|
|||||||
#include "serializer.h"
|
#include "serializer.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
struct BufferBuilder {
|
||||||
|
void* memory;
|
||||||
|
size_t size;
|
||||||
|
size_t capacity;
|
||||||
|
|
||||||
|
BufferBuilder() {
|
||||||
|
memory = malloc(128);
|
||||||
|
size = 0;
|
||||||
|
capacity = 128;
|
||||||
|
}
|
||||||
|
|
||||||
|
~BufferBuilder() {
|
||||||
|
free(memory);
|
||||||
|
size = 0;
|
||||||
|
capacity = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AppendToBuffer(void* content, size_t content_size) {
|
||||||
|
if (size + content_size > capacity) {
|
||||||
|
// Grow memory if needed.
|
||||||
|
size_t new_size = capacity * 2;
|
||||||
|
while (new_size < size + content_size)
|
||||||
|
new_size *= 2;
|
||||||
|
void* new_memory = malloc(capacity);
|
||||||
|
memcpy(new_memory, memory, size);
|
||||||
|
free(memory);
|
||||||
|
memory = new_memory;
|
||||||
|
|
||||||
|
// Append new content into memory.
|
||||||
|
memcpy((char*)memory + size, content, content_size);
|
||||||
|
size += content_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Reset() {
|
||||||
|
size = 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// JSON-encoded message that is passed across shared memory.
|
// JSON-encoded message that is passed across shared memory.
|
||||||
//
|
//
|
||||||
// Messages are funky objects. They contain potentially variable amounts of
|
// Messages are funky objects. They contain potentially variable amounts of
|
||||||
@ -85,6 +124,47 @@ struct IpcDirectionalChannel::MessageBuffer {
|
|||||||
JsonMessage* free_message() {
|
JsonMessage* free_message() {
|
||||||
return message_at_offset(metadata()->bytes_used);
|
return message_at_offset(metadata()->bytes_used);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct Iterator {
|
||||||
|
void* buffer;
|
||||||
|
size_t remaining_bytes;
|
||||||
|
|
||||||
|
Iterator(void* buffer, size_t remaining_bytes) : remaining_bytes(remaining_bytes) {}
|
||||||
|
|
||||||
|
JsonMessage* get() const {
|
||||||
|
assert(buffer);
|
||||||
|
return reinterpret_cast<JsonMessage*>(buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
JsonMessage* operator->() const {
|
||||||
|
return get();
|
||||||
|
}
|
||||||
|
|
||||||
|
Iterator operator++() const {
|
||||||
|
size_t next_message_offset = sizeof(JsonMessage) + get()->payload_size;
|
||||||
|
if (next_message_offset >= remaining_bytes) {
|
||||||
|
assert(next_message_offset == remaining_bytes);
|
||||||
|
return Iterator(nullptr, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto* next_message = (char*)buffer + next_message_offset;
|
||||||
|
return Iterator(next_message, remaining_bytes - next_message_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator==(const Iterator& other) const {
|
||||||
|
return buffer == other.buffer && remaining_bytes == other.remaining_bytes;
|
||||||
|
}
|
||||||
|
bool operator!=(const Iterator& other) const {
|
||||||
|
return !(*this == other);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Iterator begin() {
|
||||||
|
return Iterator(first_message(), metadata()->bytes_used);
|
||||||
|
}
|
||||||
|
Iterator end() {
|
||||||
|
return Iterator(nullptr, 0);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
IpcDirectionalChannel::IpcDirectionalChannel(const std::string& name) {
|
IpcDirectionalChannel::IpcDirectionalChannel(const std::string& name) {
|
||||||
@ -170,19 +250,28 @@ std::vector<std::unique_ptr<IpcMessage>> IpcDirectionalChannel::TakeMessages() {
|
|||||||
|
|
||||||
std::vector<std::unique_ptr<IpcMessage>> result;
|
std::vector<std::unique_ptr<IpcMessage>> result;
|
||||||
|
|
||||||
size_t offset = 0;
|
// TODO
|
||||||
while (remaining_bytes > 0) {
|
for (auto it = local_buffer->begin(); it != local_buffer->end(); ++it) {
|
||||||
JsonMessage* message = local_buffer->message_at_offset(offset);
|
// TODO: partial payload, maybe something like this:
|
||||||
std::cerr << "remaining_bytes=" << remaining_bytes << ", offset=" << offset << ", message->payload_size=" << message->payload_size << std::endl;
|
//
|
||||||
offset += message->payload_size;
|
// if (it->partial_id != 0) {
|
||||||
remaining_bytes -= sizeof(JsonMessage) + message->payload_size;
|
// auto* buf = CreateOrFindResizableBuffer(it->partial_id);
|
||||||
|
// buf->Append(it->payload(), it->payload_size());
|
||||||
|
// if (it->is_complete) {
|
||||||
|
// Process(buf.payload(), buff.payload_size())
|
||||||
|
// RemoveResizableBuffer(it->partial_id)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// else {
|
||||||
|
// Process(it->payload(), it->payload_size())
|
||||||
|
// }
|
||||||
|
//
|
||||||
rapidjson::Document document;
|
rapidjson::Document document;
|
||||||
document.Parse(message->payload(), message->payload_size);
|
document.Parse(it->payload(), it->payload_size);
|
||||||
bool has_error = document.HasParseError();
|
bool has_error = document.HasParseError();
|
||||||
auto error = document.GetParseError();
|
auto error = document.GetParseError();
|
||||||
|
|
||||||
std::unique_ptr<IpcMessage> base_message = IpcRegistry::instance()->Allocate(message->ipc_id);
|
std::unique_ptr<IpcMessage> base_message = IpcRegistry::instance()->Allocate(it->ipc_id);
|
||||||
base_message->Deserialize(document);
|
base_message->Deserialize(document);
|
||||||
result.emplace_back(std::move(base_message));
|
result.emplace_back(std::move(base_message));
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user