A simple SoC for education using PULP IPs. Croc includes all scripts necessary to produce a nearly finished chip in IHPs open-source 130nm technology.
As it is oriented towards education, it forgoes some configurability to increase readability of the RTL and scripts.
Croc is developed as part of the PULP project, a joint effort between ETH Zurich and the University of Bologna.
Croc was successfully taped out in Nov 2024. The chip is called MLEM, named after the sound Yoshi makes when eating a tasty fruit. MLEM was designed and prepared for tapeout by ETHZ students as a bachelor project. The exact code and scripts used for the tapeout can be seen in the frozen mlem-tapeout branch.
IMPORTANT: Update to 1.1 recommended.
Release 1.1 and newer includes a fix for the SRAMs where the A_DLY pin was tied low instead of high. The pin controls internal timings and the old version may create violations for some SRAMs.
The SoC is composed of two main parts:
- The
croc_domaincontaining a CVE2 core (a fork of Ibex), SRAM, an OBI crossbar and a few simple peripherals - The
user_domainwhere students are invited to add their own designs or other open-source designs (peripherals, accelerators...)
The main interconnect is OBI, you can find the spec online.
The various IPs of the SoC (UART, OBI, debug-module, timer...) come from other PULP repositories and are managed by Bender.
To make it easier to browse and understand, only the currently used files are included in rtl/<IP>. You may want to explore the repositories of the respective IPs to find their documentation or additional functionality, the urls are in Bender.yml.
The main SoC configurations are in rtl/croc_pkg.sv:
| Parameter | Default | Function |
|---|---|---|
HartId |
0 |
Core's Hart ID |
PulpJtagIdCode |
32'hED9_C0C50 |
Debug module ID code |
NumExternalIrqs |
4 |
Number of external interrupts into Croc domain |
BankNumWords |
512 |
Number of 32bit words in a memory bank |
NumSramBanks |
2 |
Number of memory banks |
The SRAMs are instantiated via a technology wrapper called tc_sram_impl (tc: tech_cells), the technology-independent implementation is in rtl/tech_cells_generic/tc_sram_impl.sv. A number of SRAM configurations are implemented using IHP130 SRAM memories in ihp13/tc_sram_impl.sv. If an unimplemented SRAM configuration is instantiated it will result in a tc_sram_blackbox module which can then be easily identified from the synthesis results.
Currently the only way to boot is via JTAG.
If possible, the memory map should remain compatible with Cheshire's memory map.
Further each new subordinate should occupy multiples of 4KB of the address space (32'h0000_1000).
The address map of the default configuration is as follows:
| Start Address | Stop Address | Description |
|---|---|---|
32'h0000_0000 |
32'h0004_0000 |
Debug module (JTAG) |
32'h0300_0000 |
32'h0300_1000 |
SoC control/info registers |
32'h0300_2000 |
32'h0300_3000 |
UART peripheral |
32'h0300_5000 |
32'h0300_6000 |
GPIO peripheral |
32'h0300_A000 |
32'h0300_B000 |
Timer peripheral |
32'h1000_0000 |
+SRAM_SIZE |
Memory banks (SRAM) |
32'h2000_0000 |
32'h8000_0000 |
Passthrough to user domain |
32'h2000_0000 |
32'h2000_1000 |
reserved for string formatted user ROM* |
*If people modify Croc we suggest they add a ROM at this address containing additional information
like the names of the developers, a project link or similar. This can then be written out via UART.
We ask people to format the ROM like a C string with zero termination and using ASCII encoding if feasible.
The MLEM user ROM may serve as a reference implementation.
graph LR;
Bender-->Yosys;
Yosys-->OpenRoad;
OpenRoad-->KLayout;
- Bender provides a list of SystemVerilog files
- Yosys parses, elaborates, optimizes and maps the design to the technology cells
- The netlist, constraints and floorplan are loaded into OpenRoad for Place&Route
- The design as def is read by klayout and the geometry of the cells and macros are merged
Currently, the final GDS is still missing the following things:
- metal density fill
- sealring These can be added in KLayout, check the IHP repository (possible the dev branch) for a reference script.
| Cell/Module placement | Routing |
|---|---|
![]() |
![]() |
We are using the excellent docker container maintained by Harald Pretl. If you get stuck with installing the tools, we urge you to check the Tool Repository.
The current supported version is 2025.03, no other version is officially supported.
ETHZ Design Center maintains an internal version of the IHP PDK, with integrations into all tools we have access to. For this reason if you work on the ETH systems it is recommended to use the icdesign tool (cockpit) instead of the liked Github repo.
You can directly create a cockpit directory inside the croc directory:
# Make sure you are in <somedir>/croc
# the checked-out repository
icdesign ihp13 -noguiThe setup is guided by the .cockpitrc configuration file. If you need different macros or another version of the standard cells you can change it accordingly.
An environment setup for bash is provided to get easy access to the tools:
source ethz.envAdditionally you may prefer to just enter a shell in the pre-installed osic-tools container using:
oseda bash
# older version eg: oseda -2025.03 bashNote: this has currently only been tested on Ubuntu and RHEL Linux.
There are two possible ways, the easiest way is to install docker and work in the docker container, you can follow the install guides on the Docker Website.
You do not need to manually download the container image, this will be done when running the script.
If you do not have git installed on your system, you also need to install Github Desktop and then clone this git repository.
It is a good idea to grant non-root (sudo) users access to docker, this is decribed in the Docker Article.
Finally, you can navigate to this directory, open a terminal (PowerShell in Windows) and type:
# Linux only (starts and enters docker container in shell)
./start_linux.sh
# Linux/Mac (starts VNC server on localhost:5901)
./start_vnc.sh
# Windows (starts VNC server on localhost:5901)
./start_vnc.batIf you use the VNC option, open a browser and type localhost in the address bar.
This should connect you to the VNC server, the password is abc123, then test by right-clicking somewhere, starting the terminal and typing ls.
You should see the files in this repository again.
Now you should be in an Ubuntu environment with all tools pre-installed for you.
If something does not work, refer to the upstream IIC-OSIC-Tools
You need to build/install the required tools manually:
- Bender: Dependency manager
- Yosys: Synthesis tool
- Yosys-Slang: SystemVerilog frontend for Yosys
- OpenRoad: Place & Route tool
- (Optional) Verilator: Simulator
- (Optional) Questasim/Modelsim: Simulator
The SoC is fully functional as-is and a simple software example is provided for simulation. To run the synthesis and place & route flow execute:
make checkout
make yosys
make openroad
make klayoutTo simulate you can use:
make verilatorIf you have Questasim/Modelsim, you can also run:
make vsimThe most important make targets are documented, you can list them with:
make helpTo add your own design, we recommend creating a new directory under rtl/ or put single source files (small designs) into rtl/user_domain, then go into Bender.yml and add the files in the indicated places.
This will make Bender aware of the files and any script it contains will contain your design as well.
Then re-generate the default synthesis file-list:
make yosys-flistIf you want to add an existing design and it already containts a Bender.yml in its repository, you can add it as a dependency in the Bender.yml and reading the guide below.
The dependency manager Bender is used in most pulp-platform IPs.
Usually each dependency would be in a seperate repository, each with a Bender.yml file to describe where the RTL files are, how you can use this dependency and which additional dependency it has.
In the top level repository (like this SoC) you also have a Bender.yml file but you will commonly find a Bender.lock file. It contains the resolved tree of dependencies with specific commits for each. Whenever you run a command using Bender, this is the file it uses to figure out where things are.
Below is a small guide aimed at the usecase for this project. The Bender repo has a more extensive Command Guide.
Using the command bender checkout Bender will check the lock file and download the specified commits from the repositories (usually into a hidden .bender directory).
Running bender update on the other hand will resolve the entire tree again and re-generate the lock file (you usually have to resolve some version/revision conflicts if multiple things use the same dependency).
Remember: always test everything again if you generate a new Bender.lock, it is the same as modifying RTL.
For this repository, we use a subcommand called bendor vendor together with the vendor_package section in Bender.yml.
bendor vendor can be used to Benderize arbitrary repositories with RTL in it. The dependencies are already 'checked out' into rtl/<IP>. Each file or directory from the repository is mapped to a local path in this repo.
Fixes and changes to each IPs rtl/<IP>/Bender.yml are managed by bender vendor in rtl/patches.
If you need to update a dependency or map another file you need to edit the coresponding vendor_package section in Bender.yml and then run bender vendor init. Then you might need to change rtl/<IP>/Bender.yml to list your new file in the sources.
To save a fix/change as a patch, stage it in git and then run bender vendor patch. When prompted, add a commit message (this is used as the patches file name). Finally, commit both the patch file and the new rtl/<IP>.
Note: using bender vendor in this repository to change the local versions of the IPs requires an up-to-date version of Bender, specifically it needs to include PR 179.
Another thing we use are targets (in the Bender.yml), together they build different views/contexts of your RTL. For example without defining any targets the technology independent cells/memories are used (in rtl/tech_cells_generic/) but if we use the target ihp13 then the same modules contain a technology-specific implementation (in ihp13/). Similar contexts are built for different simulators and other things.
Unless specified otherwise in the respective file headers, all code checked into this repository is made available under a permissive license. All hardware sources and tool scripts are licensed under the Solderpad Hardware License 0.51 (see LICENSE.md). All software sources are licensed under Apache 2.0.

