fuzz_util.cpp 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  1. /* This file is part of the dynarmic project.
  2. * Copyright (c) 2018 MerryMage
  3. * SPDX-License-Identifier: 0BSD
  4. */
  5. #include "./fuzz_util.h"
  6. #include <cstring>
  7. #include <fmt/format.h>
  8. #include <fmt/ostream.h>
  9. #include <mcl/assert.hpp>
  10. #include "./rand_int.h"
  11. #include "dynarmic/common/fp/fpcr.h"
  12. #include "dynarmic/common/fp/rounding_mode.h"
  13. using namespace Dynarmic;
  14. std::ostream& operator<<(std::ostream& o, Vector vec) {
  15. return o << fmt::format("{:016x}'{:016x}", vec[1], vec[0]);
  16. }
  17. Vector RandomVector() {
  18. return {RandInt<u64>(0, ~u64(0)), RandInt<u64>(0, ~u64(0))};
  19. }
  20. u32 RandomFpcr() {
  21. FP::FPCR fpcr;
  22. fpcr.AHP(RandInt(0, 1) == 0);
  23. fpcr.DN(RandInt(0, 1) == 0);
  24. fpcr.FZ(RandInt(0, 1) == 0);
  25. fpcr.RMode(static_cast<FP::RoundingMode>(RandInt(0, 3)));
  26. fpcr.FZ16(RandInt(0, 1) == 0);
  27. return fpcr.Value();
  28. }
  29. InstructionGenerator::InstructionGenerator(const char* format) {
  30. const size_t format_len = std::strlen(format);
  31. ASSERT(format_len == 16 || format_len == 32);
  32. if (format_len == 16) {
  33. // Begin with 16 zeros
  34. mask |= 0xFFFF0000;
  35. }
  36. for (size_t i = 0; i < format_len; i++) {
  37. const u32 bit = 1u << (format_len - i - 1);
  38. switch (format[i]) {
  39. case '0':
  40. mask |= bit;
  41. break;
  42. case '1':
  43. bits |= bit;
  44. mask |= bit;
  45. break;
  46. default:
  47. // Do nothing
  48. break;
  49. }
  50. }
  51. }
  52. u32 InstructionGenerator::Generate() const {
  53. const u32 random = RandInt<u32>(0, 0xFFFFFFFF);
  54. return bits | (random & ~mask);
  55. }