Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Bin format #1847

Merged
merged 26 commits into from
Oct 9, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
faa222f
create capnp folder
treelin611 Sep 25, 2024
5174b7a
add capnp for unique blocks and add write bin function
treelin611 Sep 26, 2024
0cca495
write bin format function (with bug)
treelin611 Sep 26, 2024
3fcdc10
write bin function no compile error
treelin611 Sep 27, 2024
ef18d04
write bin function works now
treelin611 Sep 27, 2024
1d6f990
renamed file
treelin611 Sep 27, 2024
59f1e4a
add read bin (with bugs)
treelin611 Sep 27, 2024
ed38169
read bin format mod (with bug)
treelin611 Sep 27, 2024
08ec376
mod read bin
treelin611 Sep 29, 2024
87ca9f3
add three testcases to test bin read and write
treelin611 Sep 29, 2024
6864159
modified testcases
treelin611 Sep 29, 2024
bddf693
read bin ready
treelin611 Sep 29, 2024
20238c1
modified testcase
treelin611 Sep 29, 2024
7d9a677
changed file name
treelin611 Sep 29, 2024
bceb160
reformat code
treelin611 Sep 29, 2024
3db116f
Merge branch 'master' into bin_format
treelin611 Oct 8, 2024
be3546f
Merge branch 'master' into bin_format
treelin611 Oct 8, 2024
1ba3298
add uxsdcxx
treelin611 Oct 8, 2024
f0a9ca8
add xsd file and modified cmakelist
treelin611 Oct 8, 2024
9131e74
modified CMakeLists.txt
treelin611 Oct 8, 2024
03ccfa1
reformat code
treelin611 Oct 8, 2024
f0a52be
auto generate capnp no compile error
treelin611 Oct 9, 2024
5b0e52a
Merge branch 'master' into bin_format
treelin611 Oct 9, 2024
4d8fae9
seperate xml parser and bin parser
treelin611 Oct 9, 2024
88e12a0
modified test cases & xsd file
treelin611 Oct 9, 2024
fc5c0f6
modified testcases
treelin611 Oct 9, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions libs/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -10,3 +10,4 @@ add_subdirectory(libpcf)
add_subdirectory(libbusgroup)
add_subdirectory(libnamemanager)
add_subdirectory(libtileconfig)
add_subdirectory(libopenfpgacapnproto)
90 changes: 90 additions & 0 deletions libs/libopenfpgacapnproto/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
include(GNUInstallDirs)

if(NOT MSCV)
# These flags generate noisy but non-bug warnings when using lib kj,
# supress them.
set(WARN_FLAGS_TO_DISABLE
-Wno-undef
-Wno-non-virtual-dtor
)
foreach(flag ${WARN_FLAGS_TO_DISABLE})
CHECK_CXX_COMPILER_FLAG(${flag} CXX_COMPILER_SUPPORTS_${flag})
if(CXX_COMPILER_SUPPORTS_${flag})
#Flag supported, so enable it
add_compile_options(${flag})
endif()
endforeach()
endif()

# Create generated headers from capnp schema files
#
# Each schema used should appear here.
set(CAPNP_DEFS
gen/unique_blocks_uxsdcxx.capnp
)

capnp_generate_cpp(CAPNP_SRCS CAPNP_HDRS
${CAPNP_DEFS}
)


set(IC_DIR ${CMAKE_CURRENT_SOURCE_DIR}/../../vtr-verilog-to-routing/libs/EXTERNAL/libinterchange/interchange)
set(CAPNPC_SRC_PREFIX ${IC_DIR})

find_program(WGET wget REQUIRED)
find_package(ZLIB REQUIRED)

# Add Java schema
set(JAVA_SCHEMA ${CMAKE_CURRENT_BINARY_DIR}/../../vtr-verilog-to-routing/libs/libvtrcapnproto/schema/capnp/java.capnp)
add_custom_command(
OUTPUT ${JAVA_SCHEMA}
COMMAND ${CMAKE_COMMAND} -E make_directory ${CMAKE_CURRENT_BINARY_DIR}/../../vtr-verilog-to-routing/libs/libvtrcapnproto/schema/capnp/
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
COMMAND ${WGET}
https://raw.githubusercontent.com/capnproto/capnproto-java/master/compiler/src/main/schema/capnp/java.capnp
-O ${JAVA_SCHEMA}
)


set(CAPNPC_IMPORT_DIRS)
list(APPEND CAPNPC_IMPORT_DIRS ${CMAKE_CURRENT_BINARY_DIR}/../../vtr-verilog-to-routing/libs/libvtrcapnproto/schema)

treelin611 marked this conversation as resolved.
Show resolved Hide resolved
set(IC_PROTOS)
set(IC_SRCS)
set(IC_HDRS)
foreach(PROTO ${IC_PROTOS})
capnp_generate_cpp(
IC_SRC
IC_HDR
${IC_DIR}/${PROTO}
)
list(APPEND IC_SRCS ${IC_SRC})
list(APPEND IC_HDRS ${IC_HDR})
list(APPEND CAPNP_DEFS ${IC_DIR}/${PROTO})
endforeach()


install(FILES ${CAPNP_DEFS} DESTINATION ${CMAKE_INSTALL_DATADIR}/openfpga)

add_library(libopenfpgacapnproto STATIC
${CAPNP_SRCS}
${IC_SRCS}
mmap_file.h
mmap_file.cpp
serdes_utils.h
serdes_utils.cpp
)


add_dependencies(libopenfpgacapnproto
get_java_capnp_schema)


target_include_directories(libopenfpgacapnproto PUBLIC
${CMAKE_CURRENT_SOURCE_DIR}
${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_BINARY_DIR}/gen
)
target_link_libraries(libopenfpgacapnproto
libopenfpgautil
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
CapnProto::capnp
)
74 changes: 74 additions & 0 deletions libs/libopenfpgacapnproto/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
Capnproto usage in Openfpga
======================

Capnproto is a data serialization framework designed for portabliity and speed.
In Openfpga, capnproto is used to provide binary formats for internal data
structures that can be computed once, and used many times. Specific examples:
- preload unique blocks

What is capnproto?
==================

capnproto can be broken down into 3 parts:
- A schema language
- A code generator
- A library

The schema language is used to define messages. Each message must have an
explcit capnproto schema, which are stored in files suffixed with ".capnp".
The capnproto documentation for how to write these schema files can be found
here: https://capnproto.org/language.html

The schema by itself is not especially useful. In order to read and write
messages defined by the schema in a target language (e.g. C++), a code
generation step is required. Capnproto provides a cmake function for this
purpose, `capnp_generate_cpp`. This generates C++ source and header files.
These source and header files combined with the capnproto C++ library, enables
C++ code to read and write the messages matching a particular schema. The C++
library API can be found here: https://capnproto.org/cxx.html

Contents of libopenfpgacapnproto
===========================

libopenfpgacapnproto should contain two elements:
- Utilities for working capnproto messages in Openfpga
- Generate source and header files of all capnproto messages used in Openfpga

I/O Utilities
-------------

Capnproto does not provide IO support, instead it works from arrays (or file
descriptors). To avoid re-writing this code, libopenfpgacapnproto provides two
utilities that should be used whenever reading or writing capnproto message to
disk. These two files are copied :
- `serdes_utils.h` provides the writeMessageToFile function - Writes a
capnproto message to disk.
- `mmap_file.h` provides MmapFile object - Maps a capnproto message from the
disk as a flat array.

Capnproto schemas
-----------------

libopenfpgacapnproto should contain all capnproto schema definitions used within
VTR. To add a new schema:
1. Add the schema to git in `libs/libopenfpgacapnproto/`
2. Add the schema file name to `capnp_generate_cpp` invocation in
`libs/libopenfpgacapnproto/CMakeLists.txt`.

The schema will be available in the header file `schema filename>.h`. The
actual header file will appear in the CMake build directory
`libs/libopenfpgacapnproto` after `libopenfpgacapnproto` has been rebuilt.

Writing capnproto binary files to text
======================================

The `capnp` tool (found in the CMake build directiory
`/vtr-verilog-to-routing/libs/EXTERNAL/capnproto/c++/src/capnp`) can be used to convert from a binary
capnp message to a textual form.

Example converting UniqueBlockCompactInfo from binary to text:

```
capnp convert binary:text unique_blocks_uxsdcxx.capnp UniqueBlockCompactInfo \
< test.bin > test.txt
```
4 changes: 4 additions & 0 deletions libs/libopenfpgacapnproto/gen/README.gen.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
`unique_blocks_uxsdcxx.capnp` is generated via uxsdcxx and is checked in to
avoid requiring python3 and the uxsdcxx depedencies to build VPR.


38 changes: 38 additions & 0 deletions libs/libopenfpgacapnproto/gen/unique_blocks_uxsdcxx.capnp
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
# This file is generated by uxsdcap 0.1.0.
# https://github.com/duck2/uxsdcxx
# Modify only if your build process doesn't involve regenerating this file.
#
# Cmdline: /home/talaeikh/uxsdcxx/uxsdcap.py /home/talaeikh/vtr-verilog-to-routing/libs/librrgraph/src/io/rr_graph.xsd
# Input file: /home/talaeikh/vtr-verilog-to-routing/libs/librrgraph/src/io/rr_graph.xsd
# md5sum of input file: 9c14a0ddd3c6bc1e690ca6abf467bae6

@0xa136dddfdd48783b;
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
using Cxx = import "/capnp/c++.capnp";
$Cxx.namespace("uniqueblockcap");

enum BlockType {
cbx @0;
cby @1;
sb @2;
}

struct BlockInfo {
type @0: BlockType;
x @1: UInt32;
y @2: UInt32;
}

struct InstanceInfo {
x @0: UInt32;
y @1: UInt32;
}


struct UniqueBlockPacked {
blockInfo @0: BlockInfo;
instanceList @1: List(InstanceInfo);
}

struct UniqueBlockCompactInfo {
atomInfo @0: List(UniqueBlockPacked);
}
39 changes: 39 additions & 0 deletions libs/libopenfpgacapnproto/mmap_file.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
#include "mmap_file.h"

#include <fcntl.h>
#include <sys/stat.h>

#include "vtr_error.h"
#include "vtr_util.h"
#ifndef _WIN32
#include <sys/mman.h>
#endif
#include <unistd.h>

#include "kj/filesystem.h"

MmapFile::MmapFile(const std::string& file) : size_(0) {
try {
auto fs = kj::newDiskFilesystem();
auto path = fs->getCurrentPath().evalNative(file);

const auto& dir = fs->getRoot();
auto stat = dir.lstat(path);
auto f = dir.openFile(path);
size_ = stat.size;
data_ = f->mmap(0, stat.size);
} catch (kj::Exception& e) {
throw vtr::VtrError(e.getDescription().cStr(), e.getFile(), e.getLine());
}
}

const kj::ArrayPtr<const ::capnp::word> MmapFile::getData() const {
if ((size_ % sizeof(::capnp::word)) != 0) {
throw vtr::VtrError(
vtr::string_fmt("size_ %d is not a multiple of capnp::word", size_),
__FILE__, __LINE__);
}

return kj::arrayPtr(reinterpret_cast<const ::capnp::word*>(data_.begin()),
size_ / sizeof(::capnp::word));
}
20 changes: 20 additions & 0 deletions libs/libopenfpgacapnproto/mmap_file.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
#ifndef MMAP_FILE_H_
#define MMAP_FILE_H_

#include <string>

#include "capnp/message.h"
#include "kj/array.h"

// Platform independent mmap, useful for reading large capnp's.
class MmapFile {
public:
explicit MmapFile(const std::string& file);
const kj::ArrayPtr<const ::capnp::word> getData() const;

private:
size_t size_;
kj::Array<const kj::byte> data_;
};

#endif /* MMAP_FILE_H_ */
23 changes: 23 additions & 0 deletions libs/libopenfpgacapnproto/serdes_utils.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
#include "serdes_utils.h"

#include <fcntl.h>
#include <unistd.h>

#include "kj/filesystem.h"
#include "vtr_error.h"

void writeMessageToFile(const std::string& file,
::capnp::MessageBuilder* builder) {
try {
auto fs = kj::newDiskFilesystem();
auto path = fs->getCurrentPath().evalNative(file);

const auto& dir = fs->getRoot();
auto f = dir.openFile(path, kj::WriteMode::CREATE | kj::WriteMode::MODIFY);
f->truncate(0);
auto f_app = kj::newFileAppender(std::move(f));
capnp::writeMessage(*f_app, *builder);
} catch (kj::Exception& e) {
throw vtr::VtrError(e.getDescription().cStr(), e.getFile(), e.getLine());
}
}
21 changes: 21 additions & 0 deletions libs/libopenfpgacapnproto/serdes_utils.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
#ifndef SERDES_UTILS_H_
#define SERDES_UTILS_H_

#include <limits>
#include <string>

#include "capnp/serialize.h"

// Platform indepedent way to file message to a file on disk.
void writeMessageToFile(const std::string& file,
::capnp::MessageBuilder* builder);

inline ::capnp::ReaderOptions default_large_capnp_opts() {
::capnp::ReaderOptions opts = ::capnp::ReaderOptions();

/* Remove traversal limit */
opts.traversalLimitInWords = std::numeric_limits<uint64_t>::max();
return opts;
}

#endif /* SERDES_UTILS_H_ */
4 changes: 3 additions & 1 deletion openfpga/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,9 @@ target_link_libraries(libopenfpga
libnamemanager
libtileconfig
libpugixml
libvpr)
libvpr
libopenfpgacapnproto
)

#Create the test executable
add_executable(openfpga ${EXEC_SOURCE})
Expand Down
12 changes: 9 additions & 3 deletions openfpga/src/annotation/device_rr_gsb.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,9 @@ std::vector<vtr::Point<size_t>> DeviceRRGSB::get_sb_unique_block_instance_coord(
sb_unique_module_id_[location_x][location_y];
if (unique_module_id_instance == unique_module_id) {
vtr::Point<size_t> instance_coord(location_x, location_y);
instance_map.push_back(instance_coord);
if (instance_coord != unique_block_coord) {
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
instance_map.push_back(instance_coord);
}
}
}
}
Expand Down Expand Up @@ -144,7 +146,9 @@ DeviceRRGSB::get_cbx_unique_block_instance_coord(
cbx_unique_module_id_[location_x][location_y];
if (unique_module_id_instance == unique_module_id) {
vtr::Point<size_t> instance_coord(location_x, location_y);
instance_map.push_back(instance_coord);
if (instance_coord != unique_block_coord) {
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
instance_map.push_back(instance_coord);
}
}
}
}
Expand Down Expand Up @@ -172,7 +176,9 @@ DeviceRRGSB::get_cby_unique_block_instance_coord(
cby_unique_module_id_[location_x][location_y];
if (unique_module_id_instance == unique_module_id) {
vtr::Point<size_t> instance_coord(location_x, location_y);
instance_map.push_back(instance_coord);
if (instance_coord != unique_block_coord) {
treelin611 marked this conversation as resolved.
Show resolved Hide resolved
instance_map.push_back(instance_coord);
}
}
}
}
Expand Down
Loading
Loading