mirror of
https://github.com/VectorCamp/vectorscan.git
synced 2025-06-28 16:41:01 +03:00
187 lines
6.5 KiB
C++
187 lines
6.5 KiB
C++
/*
|
|
* Copyright (c) 2015-2017, Intel Corporation
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* * Neither the name of Intel Corporation nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "fdr_compile.h"
|
|
#include "fdr_compile_internal.h"
|
|
#include "fdr_confirm.h"
|
|
#include "fdr_dump.h"
|
|
#include "fdr_engine_description.h"
|
|
#include "fdr_internal.h"
|
|
#include "teddy_engine_description.h"
|
|
#include "teddy_internal.h"
|
|
#include "ue2common.h"
|
|
|
|
#include <cstdio>
|
|
#include <memory>
|
|
|
|
#ifndef DUMP_SUPPORT
|
|
#error No dump support!
|
|
#endif
|
|
|
|
using namespace std;
|
|
|
|
namespace ue2 {
|
|
|
|
static
|
|
bool fdrIsTeddy(const FDR *fdr) {
|
|
assert(fdr);
|
|
u32 engine = fdr->engineID;
|
|
|
|
/* teddys don't have an fdr engine description (which is why the dump code
|
|
* is so broken). */
|
|
|
|
return !getFdrDescription(engine);
|
|
}
|
|
|
|
static
|
|
void dumpLitIndex(const FDRConfirm *fdrc, FILE *f) {
|
|
const u32 *lit_index = getConfirmLitIndex(fdrc);
|
|
u32 num_lits = 1U << fdrc->nBits;
|
|
u32 lits_used = count_if(lit_index, lit_index + num_lits,
|
|
[](u32 idx) { return idx != 0; });
|
|
|
|
fprintf(f, " load %u/%u (%0.2f%%)\n", lits_used, num_lits,
|
|
(double)lits_used / (double)(num_lits)*100);
|
|
}
|
|
|
|
static
|
|
void dumpConfirms(const void *fdr_base, u32 conf_offset, u32 num_confirms,
|
|
FILE *f) {
|
|
const u32 *conf = (const u32 *)((const char *)fdr_base + conf_offset);
|
|
for (u32 i = 0; i < num_confirms; i++) {
|
|
const auto *fdrc = (const FDRConfirm *)((const char *)conf + conf[i]);
|
|
fprintf(f, " confirm %u\n", i);
|
|
fprintf(f, " andmsk 0x%016llx\n", fdrc->andmsk);
|
|
fprintf(f, " mult 0x%016llx\n", fdrc->mult);
|
|
fprintf(f, " nbits %u\n", fdrc->nBits);
|
|
fprintf(f, " groups 0x%016llx\n", fdrc->groups);
|
|
dumpLitIndex(fdrc, f);
|
|
}
|
|
}
|
|
|
|
static
|
|
void dumpTeddyReinforced(const u8 *rmsk, const u32 num_tables, FILE *f) {
|
|
// dump reinforcement masks
|
|
for (u32 b = 0; b < num_tables; b++) {
|
|
fprintf(f, " reinforcement table for bucket %u..%u:\n",
|
|
b * 8, b * 8 + 7);
|
|
for (u32 i = 0; i <= N_CHARS; i++) {
|
|
fprintf(f, " 0x%02x: ", i);
|
|
for (u32 j = 0; j < 8; j++) {
|
|
u8 val = rmsk[b * ((N_CHARS + 1) * 8) + i * 8 + j];
|
|
for (u32 k = 0; k < 8; k++) {
|
|
fprintf(f, "%s", ((val >> k) & 0x1) ? "1" : "0");
|
|
}
|
|
fprintf(f, " ");
|
|
}
|
|
fprintf(f, "\n");
|
|
}
|
|
fprintf(f, "\n");
|
|
}
|
|
}
|
|
|
|
static
|
|
void dumpTeddyMasks(const u8 *baseMsk, u32 numMasks, u32 maskWidth, FILE *f) {
|
|
// dump nibble masks
|
|
fprintf(f, " nibble masks:\n");
|
|
for (u32 i = 0; i < numMasks * 2; i++) {
|
|
fprintf(f, " -%d%s: ", 1 + i / 2, (i % 2) ? "hi" : "lo");
|
|
for (u32 j = 0; j < 16 * maskWidth; j++) {
|
|
u8 val = baseMsk[i * 16 * maskWidth + j];
|
|
for (u32 k = 0; k < 8; k++) {
|
|
fprintf(f, "%s", ((val >> k) & 0x1) ? "1" : "0");
|
|
}
|
|
fprintf(f, " ");
|
|
}
|
|
fprintf(f, "\n");
|
|
}
|
|
fprintf(f, "\n");
|
|
}
|
|
|
|
static
|
|
void dumpTeddy(const Teddy *teddy, FILE *f) {
|
|
fprintf(f, "TEDDY: %u\n", teddy->engineID);
|
|
auto des = getTeddyDescription(teddy->engineID);
|
|
if (!des) {
|
|
fprintf(f, " <unknown engine>\n");
|
|
return;
|
|
}
|
|
|
|
fprintf(f, " masks %u\n", des->numMasks);
|
|
fprintf(f, " buckets %u\n", des->getNumBuckets());
|
|
fprintf(f, " packed %s\n", des->packed ? "true" : "false");
|
|
fprintf(f, " strings %u\n", teddy->numStrings);
|
|
fprintf(f, " size %zu bytes\n", fdrSize((const FDR *)teddy));
|
|
fprintf(f, " max length %u\n", teddy->maxStringLen);
|
|
fprintf(f, " floodoff %u (%x)\n", teddy->floodOffset,
|
|
teddy->floodOffset);
|
|
fprintf(f, "\n");
|
|
|
|
u32 maskWidth = des->getNumBuckets() / 8;
|
|
size_t headerSize = sizeof(Teddy);
|
|
size_t maskLen = des->numMasks * 16 * 2 * maskWidth;
|
|
const u8 *teddy_base = (const u8 *)teddy;
|
|
const u8 *baseMsk = teddy_base + ROUNDUP_CL(headerSize);
|
|
const u8 *rmsk = baseMsk + ROUNDUP_CL(maskLen);
|
|
dumpTeddyMasks(baseMsk, des->numMasks, maskWidth, f);
|
|
dumpTeddyReinforced(rmsk, maskWidth, f);
|
|
dumpConfirms(teddy, teddy->confOffset, des->getNumBuckets(), f);
|
|
}
|
|
|
|
static
|
|
void dumpFDR(const FDR *fdr, FILE *f) {
|
|
fprintf(f, "FDR: %u\n", fdr->engineID);
|
|
auto des = getFdrDescription(fdr->engineID);
|
|
if (!des) {
|
|
fprintf(f, " <unknown engine>\n");
|
|
return;
|
|
}
|
|
|
|
fprintf(f, " domain %u\n", fdr->domain);
|
|
fprintf(f, " stride %u\n", fdr->stride);
|
|
fprintf(f, " strings %u\n", fdr->numStrings);
|
|
fprintf(f, " size %zu bytes\n", fdrSize(fdr));
|
|
fprintf(f, " max length %u\n", fdr->maxStringLen);
|
|
fprintf(f, " floodoff %u (%x)\n", fdr->floodOffset, fdr->floodOffset);
|
|
fprintf(f, "\n");
|
|
|
|
dumpConfirms(fdr, fdr->confOffset, des->getNumBuckets(), f);
|
|
}
|
|
|
|
void fdrPrintStats(const FDR *fdr, FILE *f) {
|
|
if (fdrIsTeddy(fdr)) {
|
|
dumpTeddy((const Teddy *)fdr, f);
|
|
} else {
|
|
dumpFDR(fdr, f);
|
|
}
|
|
}
|
|
|
|
} // namespace ue2
|