📄 mysql.cc
字号:
/* * Copyright (c) 2004, 2005 * The Regents of The University of Michigan * All Rights Reserved * * This code is part of the M5 simulator, developed by Nathan Binkert, * Erik Hallnor, Steve Raasch, and Steve Reinhardt, with contributions * from Ron Dreslinski, Dave Greene, Lisa Hsu, Kevin Lim, Ali Saidi, * and Andrew Schultz. * * Permission is granted to use, copy, create derivative works and * redistribute this software and such derivative works for any * purpose, so long as the copyright notice above, this grant of * permission, and the disclaimer below appear in all copies made; and * so long as the name of The University of Michigan is not used in * any advertising or publicity pertaining to the use or distribution * of this software without specific, written prior authorization. * * THIS SOFTWARE IS PROVIDED AS IS, WITHOUT REPRESENTATION FROM THE * UNIVERSITY OF MICHIGAN AS TO ITS FITNESS FOR ANY PURPOSE, AND * WITHOUT WARRANTY BY THE UNIVERSITY OF MICHIGAN OF ANY KIND, EITHER * EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE. THE REGENTS OF THE UNIVERSITY OF MICHIGAN SHALL NOT BE * LIABLE FOR ANY DAMAGES, INCLUDING DIRECT, SPECIAL, INDIRECT, * INCIDENTAL, OR CONSEQUENTIAL DAMAGES, WITH RESPECT TO ANY CLAIM * ARISING OUT OF OR IN CONNECTION WITH THE USE OF THE SOFTWARE, EVEN * IF IT HAS BEEN OR IS HEREAFTER ADVISED OF THE POSSIBILITY OF SUCH * DAMAGES. */#include <cassert>#include <map>#include <sstream>#include <string>#include <vector>#include "base/misc.hh"#include "base/mysql.hh"#include "base/statistics.hh"#include "base/stats/flags.hh"#include "base/stats/mysql.hh"#include "base/stats/mysql_run.hh"#include "base/stats/statdb.hh"#include "base/stats/types.hh"#include "base/str.hh"#include "sim/host.hh"using namespace std;namespace Stats {MySqlRun MySqlDB;boolMySqlConnected(){ return MySqlDB.connected();}voidMySqlRun::connect(const string &host, const string &user, const string &passwd, const string &db, const string &name, const string &sample, const string &project){ if (connected()) panic("can only get one database connection at this time!"); mysql.connect(host, user, passwd, db); if (mysql.error) panic("could not connect to database server\n%s\n", mysql.error); if (mysql.autocommit(false)) panic("could not set autocommit\n"); remove(name); //cleanup(); setup(name, sample, user, project); if (mysql.commit()) panic("could not commit transaction\n");}voidMySqlRun::setup(const string &name, const string &sample, const string &user, const string &project){ assert(mysql.connected()); stringstream insert; ccprintf(insert, "INSERT INTO " "runs(rn_name,rn_sample,rn_user,rn_project,rn_date,rn_expire)" "values(\"%s\", \"%s\", \"%s\", \"%s\", NOW()," "DATE_ADD(CURDATE(), INTERVAL 31 DAY))", name, sample, user, project); mysql.query(insert); if (mysql.error) panic("could not get a run\n%s\n", mysql.error); run_id = mysql.insert_id();}voidMySqlRun::remove(const string &name){ assert(mysql.connected()); stringstream sql; ccprintf(sql, "DELETE FROM runs WHERE rn_name=\"%s\"", name); mysql.query(sql); if (mysql.error) panic("could not delete run\n%s\n", mysql.error);}voidMySqlRun::cleanup(){ assert(mysql.connected()); mysql.query("DELETE data " "FROM data " "LEFT JOIN runs ON dt_run=rn_id " "WHERE rn_id IS NULL"); mysql.query("DELETE formula_ref " "FROM formula_ref " "LEFT JOIN runs ON fr_run=rn_id " "WHERE rn_id IS NULL"); mysql.query("DELETE formulas " "FROM formulas " "LEFT JOIN formula_ref ON fm_stat=fr_stat " "WHERE fr_stat IS NULL"); mysql.query("DELETE stats " "FROM stats " "LEFT JOIN data ON st_id=dt_stat " "WHERE dt_stat IS NULL"); mysql.query("DELETE subdata " "FROM subdata " "LEFT JOIN data ON sd_stat=dt_stat " "WHERE dt_stat IS NULL"); mysql.query("DELETE bins " "FROM bins " "LEFT JOIN data ON bn_id=dt_bin " "WHERE dt_bin IS NULL"); mysql.query("DELETE events" "FROM events" "LEFT JOIN runs ON ev_run=rn_id" "WHERE rn_id IS NULL"); mysql.query("DELETE event_names" "FROM event_names" "LEFT JOIN events ON en_id=ev_event" "WHERE ev_event IS NULL");}voidSetupStat::init(){ name = ""; descr = ""; type = ""; print = false; prereq = 0; prec = -1; nozero = false; nonan = false; total = false; pdf = false; cdf = false; min = 0; max = 0; bktsize = 0; size = 0;}unsignedSetupStat::setup(){ MySQL::Connection &mysql = MySqlDB.conn(); stringstream insert; ccprintf(insert, "INSERT INTO " "stats(st_name, st_descr, st_type, st_print, st_prereq, " "st_prec, st_nozero, st_nonan, st_total, st_pdf, st_cdf, " "st_min, st_max, st_bktsize, st_size)" "values(\"%s\",\"%s\",\"%s\"," " %d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d)", name, descr, type, print, prereq, (int)prec, nozero, nonan, total, pdf, cdf, min, max, bktsize, size); mysql.query(insert); if (!mysql.error) return mysql.insert_id(); stringstream select; ccprintf(select, "SELECT * FROM stats WHERE st_name=\"%s\"", name); mysql.query(select); MySQL::Result result = mysql.store_result(); if (!result) panic("could not find stat\n%s\n", mysql.error); if (mysql.commit()) panic("could not commit transaction\n"); assert(result.num_fields() == 16); MySQL::Row row = result.fetch_row(); if (!row) panic("could not get stat row\n%s\n", mysql.error); bool tb; int8_t ti8; uint16_t tu16; int64_t ti64; uint64_t tu64; if (name != (char *)row[1]) panic("failed stat check on %s:name. %s != %s\n", name, name, row[1]); if (descr != (char *)row[2]) panic("failed stat check on %s:descr. %s != %s\n", name, descr, row[2]); if (type != (char *)row[3]) panic("failed stat check on %s:type. %s != %s\n", name, type, row[3]); if (!to_number(row[4], tb) || print != tb) panic("failed stat check on %s:print. %d != %d\n", name, print, tb); if (!to_number(row[6], ti8) || prec != ti8) panic("failed stat check on %s:prec. %d != %d\n", name, prec, ti8); if (!to_number(row[7], tb) || nozero != tb) panic("failed stat check on %s:nozero. %d != %d\n", name, nozero, tb); if (!to_number(row[8], tb) || nonan != tb) panic("failed stat check on %s:nonan. %d != %d\n", name, nonan, tb); if (!to_number(row[9], tb) || total != tb) panic("failed stat check on %s:total. %d != %d\n", name, total, tb); if (!to_number(row[10], tb) || pdf != tb) panic("failed stat check on %s:pdf. %d != %d\n", name, pdf, tb); if (!to_number(row[11], tb) || cdf != tb) panic("failed stat check on %s:cdf. %d != %d\n", name, cdf, tb); if (!to_number(row[12], ti64) || min != ti64) panic("failed stat check on %s:min. %d != %d\n", name, min, ti64); if (!to_number(row[13], ti64) || max != ti64) panic("failed stat check on %s:max. %d != %d\n", name, max, ti64); if (!to_number(row[14], tu64) || bktsize != tu64) panic("failed stat check on %s:bktsize. %d != %d\n", name, bktsize, tu64); if (!to_number(row[15], tu16) || size != tu16) panic("failed stat check on %s:size. %d != %d\n", name, size, tu16); to_number(row[5], prereq); uint16_t statid; to_number(row[0], statid); return statid;}unsignedSetupBin(const string &bin){ static map<string, int> binmap; using namespace MySQL; map<string,int>::const_iterator i = binmap.find(bin); if (i != binmap.end()) return (*i).second; Connection &mysql = MySqlDB.conn(); assert(mysql.connected()); if (mysql.commit()) panic("could not commit transaction\n"); uint16_t bin_id; stringstream select; stringstream insert; ccprintf(select, "SELECT bn_id FROM bins WHERE bn_name=\"%s\"", bin); mysql.query(select); MySQL::Result result = mysql.store_result(); if (result) { assert(result.num_fields() == 1); MySQL::Row row = result.fetch_row(); if (row) { to_number(row[0], bin_id); goto exit; } } ccprintf(insert, "INSERT INTO bins(bn_name) values(\"%s\")", bin); mysql.query(insert); if (mysql.error) panic("could not get a bin\n%s\n", mysql.error); bin_id = mysql.insert_id(); binmap.insert(make_pair(bin, bin_id)); exit: if (mysql.commit()) panic("could not commit transaction\n"); return bin_id;}InsertData::InsertData(){ query = new char[maxsize + 1]; size = 0; flush();}InsertData::~InsertData(){ delete [] query;}voidInsertData::flush(){ if (size) { MySQL::Connection &mysql = MySqlDB.conn(); assert(mysql.connected()); mysql.query(query); if (mysql.error) panic("could not insert data\n%s\n", mysql.error); if (mysql.commit()) panic("could not commit transaction\n"); } query[0] = '\0'; size = 0; first = true; strcpy(query, "INSERT INTO " "data(dt_stat,dt_x,dt_y,dt_run,dt_tick,dt_bin,dt_data) " "values"); size = strlen(query);}voidInsertData::insert(){ if (size + 1024 > maxsize) flush(); if (!first) { query[size++] = ','; query[size] = '\0'; } first = false; size += sprintf(query + size, "(%u,%d,%d,%u,%llu,%u,\"%f\")", stat, x, y, MySqlDB.run(), (unsigned long long)tick, bin, data);}struct InsertSubData{ uint16_t stat; int16_t x; int16_t y; string name; string descr; void setup();};voidInsertSubData::setup(){ MySQL::Connection &mysql = MySqlDB.conn(); assert(mysql.connected()); stringstream insert; ccprintf(insert, "INSERT INTO subdata(sd_stat,sd_x,sd_y,sd_name,sd_descr) " "values(%d,%d,%d,\"%s\",\"%s\")", stat, x, y, name, descr); mysql.query(insert);// if (mysql.error)// panic("could not insert subdata\n%s\n", mysql.error); if (mysql.commit()) panic("could not commit transaction\n");}voidInsertFormula(uint16_t stat, const string &formula){ MySQL::Connection &mysql = MySqlDB.conn(); assert(mysql.connected()); stringstream insert_formula; ccprintf(insert_formula, "INSERT INTO formulas(fm_stat,fm_formula) values(%d, \"%s\")", stat, formula); mysql.query(insert_formula);// if (mysql.error)// panic("could not insert formula\n%s\n", mysql.error); stringstream insert_ref; ccprintf(insert_ref, "INSERT INTO formula_ref(fr_stat,fr_run) values(%d, %d)", stat, MySqlDB.run()); mysql.query(insert_ref);// if (mysql.error)// panic("could not insert formula reference\n%s\n", mysql.error); if (mysql.commit())
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -