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

Add Instruction Test #120

Draft
wants to merge 2 commits into
base: master
Choose a base branch
from
Draft

Add Instruction Test #120

wants to merge 2 commits into from

Conversation

trdthg
Copy link
Contributor

@trdthg trdthg commented Apr 18, 2024

Related to #7

This demo currently adds some test cases for instruction_to_str(), and also includes a demo function to generate test cases

  • Most of the test cases are generated automatically, one for each instruction (except for the first 7, which I first tried to write by hand, but it was too inefficient).
  • Registers, Imm, addresses are simply set to 1 (or other values as needed)

@jdupak

  • Do you have any suggestions on the method I'm using here, or provide more test topics?
  • What about programloader tests, should we now simply hand write tests to cover as many instructions and section types as possible, or look for ways to reuse the sail-riscv test set? (The latter seems pretty complicated.)

Copy link
Collaborator

@jdupak jdupak left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Amazing work so far. I only have some organizational comments.

// QCOMPARE(Instruction(0x4432146), Instruction(1, 2, 3, 4, 5, 6));
// QCOMPARE(Instruction(0x4430004), Instruction(1, 2, 3, 4));
// QCOMPARE(Instruction(0x4000002), Instruction(1, 2_addr));
QCOMPARE(Instruction(0x0), Instruction(Instruction(0x0)));
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am confused by Instruction(Instruction(...)). What is it supposed to mean?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

just for this constructor

Instruction::Instruction(const Instruction &i) {
    this->dt = i.data();
}

indeed seems useless, I'll delete it.

@@ -1514,3 +1514,133 @@ TokenizedInstruction::TokenizedInstruction(
, address(address)
, filename(std::move(filename))
, line(line) {}


void gen_instruction_code_and_expected_str(QTextStream& outfile, const InstructionMap* im, uint32_t code, int count, int offset, int subclass_i, int depth) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should be in the instruction.test.cpp file. It has no use during normal operation of the simulator. Also I would prefer the name "generate_code_and_string_data" and a little docstring explaining the usage.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

agree


QString test_code = QString::asprintf("QCOMPARE(Instruction(0x%x).to_str(), \"%s\");", code, qPrintable(res));
outfile << test_code;
if (Instruction(code).to_str() != res) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am not sure why this is here. I would expect the test generation and execution to be separate.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should it be in a separate file as an executable? (for generating test cases)

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sounds great.

#include <iostream>
#include <fstream>
#include <qfile.h>
void gen() {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you make the name more descriptive? Also, there should be an empty line after the includes.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also move it to the test file.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same as above

@jdupak
Copy link
Collaborator

jdupak commented Apr 18, 2024

I will address the other questions shortly.

@jdupak
Copy link
Collaborator

jdupak commented Apr 18, 2024

What about programloader tests, should we now simply hand write tests to cover as many instructions and section types as possible, or look for ways to reuse the sail-riscv test set? (The latter seems pretty complicated.)

I don't think we need to dig into the instructions here. We need to test the loading functionality. I think it would be nice to come up with one executable with various sections and then "randomly" check the contents of the memory.
As for sail-riscv, as far as I understand, they are just binaries of the official risc-v test, which we are running already (see the CI).

@jdupak
Copy link
Collaborator

jdupak commented Apr 18, 2024

Do you have any suggestions on the method I'm using here, or provide more test topics?

I like your approach. Since you have the pairs already generated, I would love to add also the other direction (code_from_string).

After that, there are many commented-out MIPS tests in the core.test.cpp. Converting them to RISC-V would be great.

@jdupak
Copy link
Collaborator

jdupak commented Apr 18, 2024

We will be able to close #7 once all QSKIP calls are gone.

@trdthg trdthg force-pushed the issue7 branch 2 times, most recently from e77d427 to 1e6e49f Compare April 25, 2024 09:15
@trdthg trdthg changed the title add test Add Instruction Test Apr 25, 2024
@trdthg trdthg marked this pull request as ready for review April 25, 2024 18:42
@jdupak jdupak marked this pull request as draft May 2, 2024 09:29
@jdupak
Copy link
Collaborator

jdupak commented May 2, 2024

Good work so far. Some notes.

  • You will need to clean up the commits a bit.
  • Try making the inputs more random. x1 and immediate zero is not so great test.

@trdthg
Copy link
Contributor Author

trdthg commented Jun 26, 2024

Completely refactored

Now instruction.test.gendata.cpp iterates through all the instructions on instruction_map, tries to generate 100 (code and string_data)pair for each, then checks to_str and code_from_string and fails if they don't match.

The current generation logic relies on arg_desc's min and max.

This is an executable file, the output looks like this:

...
inst: [jalr] 

  - [ 0]    code: [  fb268267] str: [             jalr x4, -78(x13)]
  - [ 1]    code: [  7dce8a67] str: [           jalr x20, 2012(x29)]
  - [ 2]    code: [  cfc78ee7] str: [           jalr x29, -772(x15)]
  ...

inst: [j] 

  - [ 0]    code: [  e32f006f] str: [                     j -0xf9ce]
  ...
...

Since it can't generate a fixed test case every time you run this script, the file name shouldn't be gendata anymore. I tried to move the code from main to instruction.test.cpp, but fails.

Because there are too many structures used here that instruction.cpp doesn't expose (e.g. ArgumentDesc, C_inst_map, fill_argdesbycode ...), So I include instruction.cpp directly.

@jdupak could you give me some advise?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants