Skip to content

Hippo/HippoCafe

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

hippocafe

A java bytecode manipulation library written in C++.

Building

mkdir build
cd build
cmake ..
cmake --build . --config Release
cmake --install . --config Release

Example Usage

#include <hippo/cafe.hpp>

using namespace cafe;

// The source of the class file, can either be a std::vector<int8_t> or std::istream.
std::ifstream source("data/HelloWorld.class", std::ios::binary);

class_reader reader;
result<class_file> read_result = reader.read(source);
if (!read_result) {
  std::cerr << read_result.error().message() << std::endl;
  return;
}
class_file& file = read_result.value();

for (auto& method : file.methods) {
  for (auto& insn : method.code) {
    if (auto* push = std::get_if<push_insn>(&insn)) {
      if (push->value == value{"Hello World"}) {
        push->value = "Modified";
      }
    }
  }
}

class_writer writer;
std::vector<int8_t> data = writer.write(file);
std::ofstream os("HelloWorld.class", std::ios::binary);
os.write(reinterpret_cast<const char*>(data.data()), data.size());

Class Writer Flags

hippocafe's class writer can be configured to automatically compute the method maxes (max stack and max locals) and/or stack map frames. In order to compute maxes, simply pass cafe::compute_maxes to your cafe::class_writer

cafe::class_writer writer(cafe::compute_maxes);

To compute frames you must create a cafe::class_tree, then pass it to the class writer as such.

cafe::class_tree tree(cafe::load_rt);
cafe::class_writer writer(tree);

For more accurate frame generation, you can add class information to the class tree.

tree.put("my/ThisClass", "my/SuperClass", {"my/Interface1", "my/Interface2"});
// or
cafe::class_file file = ...;
tree.put(file);