123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929 |
- /*
- * PMBus wrapper over SMBus
- *
- * Copyright 2021 Google LLC
- *
- * SPDX-License-Identifier: GPL-2.0-or-later
- */
- #include "qemu/osdep.h"
- #include <math.h>
- #include "hw/i2c/pmbus_device.h"
- #include "migration/vmstate.h"
- #include "qemu/module.h"
- #include "qemu/log.h"
- uint16_t pmbus_data2direct_mode(PMBusCoefficients c, uint32_t value)
- {
- /* R is usually negative to fit large readings into 16 bits */
- uint16_t y = (c.m * value + c.b) * pow(10, c.R);
- return y;
- }
- uint32_t pmbus_direct_mode2data(PMBusCoefficients c, uint16_t value)
- {
- /* X = (Y * 10^-R - b) / m */
- uint32_t x = (value / pow(10, c.R) - c.b) / c.m;
- return x;
- }
- uint16_t pmbus_data2linear_mode(uint16_t value, int exp)
- {
- /* L = D * 2^(-e) */
- if (exp < 0) {
- return value << (-exp);
- }
- return value >> exp;
- }
- uint16_t pmbus_linear_mode2data(uint16_t value, int exp)
- {
- /* D = L * 2^e */
- if (exp < 0) {
- return value >> (-exp);
- }
- return value << exp;
- }
- void pmbus_send(PMBusDevice *pmdev, const uint8_t *data, uint16_t len)
- {
- if (pmdev->out_buf_len + len > SMBUS_DATA_MAX_LEN) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "PMBus device tried to send too much data");
- len = 0;
- }
- for (int i = len - 1; i >= 0; i--) {
- pmdev->out_buf[i + pmdev->out_buf_len] = data[len - i - 1];
- }
- pmdev->out_buf_len += len;
- }
- /* Internal only, convert unsigned ints to the little endian bus */
- static void pmbus_send_uint(PMBusDevice *pmdev, uint64_t data, uint8_t size)
- {
- uint8_t bytes[8];
- g_assert(size <= 8);
- for (int i = 0; i < size; i++) {
- bytes[i] = data & 0xFF;
- data = data >> 8;
- }
- pmbus_send(pmdev, bytes, size);
- }
- void pmbus_send8(PMBusDevice *pmdev, uint8_t data)
- {
- pmbus_send_uint(pmdev, data, 1);
- }
- void pmbus_send16(PMBusDevice *pmdev, uint16_t data)
- {
- pmbus_send_uint(pmdev, data, 2);
- }
- void pmbus_send32(PMBusDevice *pmdev, uint32_t data)
- {
- pmbus_send_uint(pmdev, data, 4);
- }
- void pmbus_send64(PMBusDevice *pmdev, uint64_t data)
- {
- pmbus_send_uint(pmdev, data, 8);
- }
- void pmbus_send_string(PMBusDevice *pmdev, const char *data)
- {
- if (!data) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: %s: uninitialised read from 0x%02x\n",
- __func__, DEVICE(pmdev)->canonical_path, pmdev->code);
- return;
- }
- size_t len = strlen(data);
- g_assert(len + pmdev->out_buf_len < SMBUS_DATA_MAX_LEN);
- pmdev->out_buf[len + pmdev->out_buf_len] = len;
- for (int i = len - 1; i >= 0; i--) {
- pmdev->out_buf[i + pmdev->out_buf_len] = data[len - 1 - i];
- }
- pmdev->out_buf_len += len + 1;
- }
- uint8_t pmbus_receive_block(PMBusDevice *pmdev, uint8_t *dest, size_t len)
- {
- /* dest may contain data from previous writes */
- memset(dest, 0, len);
- /* Exclude command code from return value */
- pmdev->in_buf++;
- pmdev->in_buf_len--;
- /* The byte after the command code denotes the length */
- uint8_t sent_len = pmdev->in_buf[0];
- if (sent_len != pmdev->in_buf_len - 1) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: length mismatch. Expected %d bytes, got %d bytes\n",
- __func__, sent_len, pmdev->in_buf_len - 1);
- }
- /* exclude length byte */
- pmdev->in_buf++;
- pmdev->in_buf_len--;
- if (pmdev->in_buf_len < len) {
- len = pmdev->in_buf_len;
- }
- memcpy(dest, pmdev->in_buf, len);
- return len;
- }
- static uint64_t pmbus_receive_uint(PMBusDevice *pmdev)
- {
- uint64_t ret = 0;
- /* Exclude command code from return value */
- pmdev->in_buf++;
- pmdev->in_buf_len--;
- for (int i = pmdev->in_buf_len - 1; i >= 0; i--) {
- ret = ret << 8 | pmdev->in_buf[i];
- }
- return ret;
- }
- uint8_t pmbus_receive8(PMBusDevice *pmdev)
- {
- if (pmdev->in_buf_len - 1 != 1) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: length mismatch. Expected 1 byte, got %d bytes\n",
- __func__, pmdev->in_buf_len - 1);
- }
- return pmbus_receive_uint(pmdev);
- }
- uint16_t pmbus_receive16(PMBusDevice *pmdev)
- {
- if (pmdev->in_buf_len - 1 != 2) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: length mismatch. Expected 2 bytes, got %d bytes\n",
- __func__, pmdev->in_buf_len - 1);
- }
- return pmbus_receive_uint(pmdev);
- }
- uint32_t pmbus_receive32(PMBusDevice *pmdev)
- {
- if (pmdev->in_buf_len - 1 != 4) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: length mismatch. Expected 4 bytes, got %d bytes\n",
- __func__, pmdev->in_buf_len - 1);
- }
- return pmbus_receive_uint(pmdev);
- }
- uint64_t pmbus_receive64(PMBusDevice *pmdev)
- {
- if (pmdev->in_buf_len - 1 != 8) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: length mismatch. Expected 8 bytes, got %d bytes\n",
- __func__, pmdev->in_buf_len - 1);
- }
- return pmbus_receive_uint(pmdev);
- }
- static uint8_t pmbus_out_buf_pop(PMBusDevice *pmdev)
- {
- if (pmdev->out_buf_len == 0) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: tried to read from empty buffer",
- __func__);
- return PMBUS_ERR_BYTE;
- }
- uint8_t data = pmdev->out_buf[pmdev->out_buf_len - 1];
- pmdev->out_buf_len--;
- return data;
- }
- static void pmbus_quick_cmd(SMBusDevice *smd, uint8_t read)
- {
- PMBusDevice *pmdev = PMBUS_DEVICE(smd);
- PMBusDeviceClass *pmdc = PMBUS_DEVICE_GET_CLASS(pmdev);
- if (pmdc->quick_cmd) {
- pmdc->quick_cmd(pmdev, read);
- }
- }
- static uint8_t pmbus_pages_num(PMBusDevice *pmdev)
- {
- const PMBusDeviceClass *k = PMBUS_DEVICE_GET_CLASS(pmdev);
- /* some PMBus devices don't use the PAGE command, so they get 1 page */
- return k->device_num_pages ? : 1;
- }
- static void pmbus_pages_alloc(PMBusDevice *pmdev)
- {
- pmdev->num_pages = pmbus_pages_num(pmdev);
- pmdev->pages = g_new0(PMBusPage, pmdev->num_pages);
- }
- void pmbus_check_limits(PMBusDevice *pmdev)
- {
- for (int i = 0; i < pmdev->num_pages; i++) {
- if ((pmdev->pages[i].operation & PB_OP_ON) == 0) {
- continue; /* don't check powered off devices */
- }
- if (pmdev->pages[i].read_vout > pmdev->pages[i].vout_ov_fault_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_VOUT;
- pmdev->pages[i].status_vout |= PB_STATUS_VOUT_OV_FAULT;
- }
- if (pmdev->pages[i].read_vout > pmdev->pages[i].vout_ov_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_VOUT;
- pmdev->pages[i].status_vout |= PB_STATUS_VOUT_OV_WARN;
- }
- if (pmdev->pages[i].read_vout < pmdev->pages[i].vout_uv_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_VOUT;
- pmdev->pages[i].status_vout |= PB_STATUS_VOUT_UV_WARN;
- }
- if (pmdev->pages[i].read_vout < pmdev->pages[i].vout_uv_fault_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_VOUT;
- pmdev->pages[i].status_vout |= PB_STATUS_VOUT_UV_FAULT;
- }
- if (pmdev->pages[i].read_vin > pmdev->pages[i].vin_ov_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_INPUT;
- pmdev->pages[i].status_input |= PB_STATUS_INPUT_VIN_OV_WARN;
- }
- if (pmdev->pages[i].read_vin < pmdev->pages[i].vin_uv_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_INPUT;
- pmdev->pages[i].status_input |= PB_STATUS_INPUT_VIN_UV_WARN;
- }
- if (pmdev->pages[i].read_iout > pmdev->pages[i].iout_oc_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_IOUT_POUT;
- pmdev->pages[i].status_iout |= PB_STATUS_IOUT_OC_WARN;
- }
- if (pmdev->pages[i].read_iout > pmdev->pages[i].iout_oc_fault_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_IOUT_POUT;
- pmdev->pages[i].status_iout |= PB_STATUS_IOUT_OC_FAULT;
- }
- if (pmdev->pages[i].read_pin > pmdev->pages[i].pin_op_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_INPUT;
- pmdev->pages[i].status_input |= PB_STATUS_INPUT_PIN_OP_WARN;
- }
- if (pmdev->pages[i].read_temperature_1
- > pmdev->pages[i].ot_fault_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_TEMPERATURE;
- pmdev->pages[i].status_temperature |= PB_STATUS_OT_FAULT;
- }
- if (pmdev->pages[i].read_temperature_1
- > pmdev->pages[i].ot_warn_limit) {
- pmdev->pages[i].status_word |= PB_STATUS_TEMPERATURE;
- pmdev->pages[i].status_temperature |= PB_STATUS_OT_WARN;
- }
- }
- }
- void pmbus_idle(PMBusDevice *pmdev)
- {
- pmdev->code = PMBUS_IDLE_STATE;
- }
- /* assert the status_cml error upon receipt of malformed command */
- static void pmbus_cml_error(PMBusDevice *pmdev)
- {
- for (int i = 0; i < pmdev->num_pages; i++) {
- pmdev->pages[i].status_word |= PMBUS_STATUS_CML;
- pmdev->pages[i].status_cml |= PB_CML_FAULT_INVALID_CMD;
- }
- }
- static uint8_t pmbus_receive_byte(SMBusDevice *smd)
- {
- PMBusDevice *pmdev = PMBUS_DEVICE(smd);
- PMBusDeviceClass *pmdc = PMBUS_DEVICE_GET_CLASS(pmdev);
- uint8_t ret = PMBUS_ERR_BYTE;
- uint8_t index;
- if (pmdev->out_buf_len != 0) {
- ret = pmbus_out_buf_pop(pmdev);
- return ret;
- }
- /*
- * Reading from all pages will return the value from page 0,
- * means that all subsequent commands are to be applied to all output.
- */
- if (pmdev->page == PB_ALL_PAGES) {
- index = 0;
- } else if (pmdev->page > pmdev->num_pages - 1) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: page %d is out of range\n",
- __func__, pmdev->page);
- pmbus_cml_error(pmdev);
- return PMBUS_ERR_BYTE;
- } else {
- index = pmdev->page;
- }
- switch (pmdev->code) {
- case PMBUS_PAGE:
- pmbus_send8(pmdev, pmdev->page);
- break;
- case PMBUS_OPERATION: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].operation);
- break;
- case PMBUS_ON_OFF_CONFIG: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].on_off_config);
- break;
- case PMBUS_PHASE: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].phase);
- break;
- case PMBUS_WRITE_PROTECT: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].write_protect);
- break;
- case PMBUS_CAPABILITY:
- pmbus_send8(pmdev, pmdev->capability);
- if (pmdev->capability & BIT(7)) {
- qemu_log_mask(LOG_UNIMP,
- "%s: PEC is enabled but not yet supported.\n",
- __func__);
- }
- break;
- case PMBUS_VOUT_MODE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MODE) {
- pmbus_send8(pmdev, pmdev->pages[index].vout_mode);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_COMMAND: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_command);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_TRIM: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_trim);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_CAL_OFFSET: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_cal_offset);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_MAX: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_MARGIN_HIGH: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MARGIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_margin_high);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_MARGIN_LOW: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MARGIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_margin_low);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_TRANSITION_RATE: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_transition_rate);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_DROOP: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_droop);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_SCALE_LOOP: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_scale_loop);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_SCALE_MONITOR: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_scale_monitor);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_MIN: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_min);
- } else {
- goto passthough;
- }
- break;
- /* TODO: implement coefficients support */
- case PMBUS_POUT_MAX: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT) {
- pmbus_send16(pmdev, pmdev->pages[index].pout_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_ON: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_on);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_OFF: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_off);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_CAL_GAIN: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT_GAIN) {
- pmbus_send16(pmdev, pmdev->pages[index].iout_cal_gain);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_CONFIG_1_2: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].fan_config_1_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_COMMAND_1: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].fan_command_1);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_COMMAND_2: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].fan_command_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_CONFIG_3_4: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].fan_config_3_4);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_COMMAND_3: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].fan_command_3);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_FAN_COMMAND_4: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].fan_command_4);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_OV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_ov_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_OV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].vout_ov_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_OV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_ov_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_UV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_uv_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_UV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].vout_uv_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VOUT_UV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].vout_uv_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_OC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].iout_oc_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_OC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].iout_oc_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_OC_LV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].iout_oc_lv_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_OC_LV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].iout_oc_lv_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_OC_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].iout_oc_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_UC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].iout_uc_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IOUT_UC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].iout_uc_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_OT_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send16(pmdev, pmdev->pages[index].ot_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_OT_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send8(pmdev, pmdev->pages[index].ot_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_OT_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send16(pmdev, pmdev->pages[index].ot_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_UT_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send16(pmdev, pmdev->pages[index].ut_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_UT_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send16(pmdev, pmdev->pages[index].ut_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_UT_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send8(pmdev, pmdev->pages[index].ut_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_OV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_ov_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_OV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send8(pmdev, pmdev->pages[index].vin_ov_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_OV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_ov_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_UV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_uv_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_UV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].vin_uv_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_VIN_UV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send8(pmdev, pmdev->pages[index].vin_uv_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IIN_OC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmbus_send16(pmdev, pmdev->pages[index].iin_oc_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IIN_OC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmbus_send8(pmdev, pmdev->pages[index].iin_oc_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IIN_OC_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmbus_send16(pmdev, pmdev->pages[index].iin_oc_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_POUT_OP_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT) {
- pmbus_send16(pmdev, pmdev->pages[index].pout_op_fault_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_POUT_OP_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT) {
- pmbus_send8(pmdev, pmdev->pages[index].pout_op_fault_response);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_POUT_OP_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT) {
- pmbus_send16(pmdev, pmdev->pages[index].pout_op_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_PIN_OP_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_PIN) {
- pmbus_send16(pmdev, pmdev->pages[index].pin_op_warn_limit);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_BYTE: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].status_word & 0xFF);
- break;
- case PMBUS_STATUS_WORD: /* R/W word */
- pmbus_send16(pmdev, pmdev->pages[index].status_word);
- break;
- case PMBUS_STATUS_VOUT: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].status_vout);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_IOUT: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send8(pmdev, pmdev->pages[index].status_iout);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_INPUT: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN ||
- pmdev->pages[index].page_flags & PB_HAS_IIN ||
- pmdev->pages[index].page_flags & PB_HAS_PIN) {
- pmbus_send8(pmdev, pmdev->pages[index].status_input);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_TEMPERATURE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send8(pmdev, pmdev->pages[index].status_temperature);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_CML: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].status_cml);
- break;
- case PMBUS_STATUS_OTHER: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].status_other);
- break;
- case PMBUS_STATUS_MFR_SPECIFIC: /* R/W byte */
- pmbus_send8(pmdev, pmdev->pages[index].status_mfr_specific);
- break;
- case PMBUS_STATUS_FANS_1_2: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].status_fans_1_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_STATUS_FANS_3_4: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].status_fans_3_4);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_EIN: /* Read-Only block 5 bytes */
- if (pmdev->pages[index].page_flags & PB_HAS_EIN) {
- pmbus_send(pmdev, pmdev->pages[index].read_ein, 5);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_EOUT: /* Read-Only block 5 bytes */
- if (pmdev->pages[index].page_flags & PB_HAS_EOUT) {
- pmbus_send(pmdev, pmdev->pages[index].read_eout, 5);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_VIN: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_vin);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_IIN: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_iin);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_VCAP: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VCAP) {
- pmbus_send16(pmdev, pmdev->pages[index].read_vcap);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_VOUT: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].read_vout);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_IOUT: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmbus_send16(pmdev, pmdev->pages[index].read_iout);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_TEMPERATURE_1: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmbus_send16(pmdev, pmdev->pages[index].read_temperature_1);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_TEMPERATURE_2: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMP2) {
- pmbus_send16(pmdev, pmdev->pages[index].read_temperature_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_TEMPERATURE_3: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMP3) {
- pmbus_send16(pmdev, pmdev->pages[index].read_temperature_3);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_FAN_SPEED_1: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_fan_speed_1);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_FAN_SPEED_2: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_fan_speed_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_FAN_SPEED_3: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_fan_speed_3);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_FAN_SPEED_4: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_fan_speed_4);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_DUTY_CYCLE: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_duty_cycle);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_FREQUENCY: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_frequency);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_POUT: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT) {
- pmbus_send16(pmdev, pmdev->pages[index].read_pout);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_READ_PIN: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_PIN) {
- pmbus_send16(pmdev, pmdev->pages[index].read_pin);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_REVISION: /* Read-Only byte */
- pmbus_send8(pmdev, pmdev->pages[index].revision);
- break;
- case PMBUS_MFR_ID: /* R/W block */
- if (pmdev->pages[index].page_flags & PB_HAS_MFR_INFO) {
- pmbus_send_string(pmdev, pmdev->pages[index].mfr_id);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_MODEL: /* R/W block */
- if (pmdev->pages[index].page_flags & PB_HAS_MFR_INFO) {
- pmbus_send_string(pmdev, pmdev->pages[index].mfr_model);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_REVISION: /* R/W block */
- if (pmdev->pages[index].page_flags & PB_HAS_MFR_INFO) {
- pmbus_send_string(pmdev, pmdev->pages[index].mfr_revision);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_LOCATION: /* R/W block */
- if (pmdev->pages[index].page_flags & PB_HAS_MFR_INFO) {
- pmbus_send_string(pmdev, pmdev->pages[index].mfr_location);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_VIN_MIN: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_vin_min);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_VIN_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_vin_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_IIN_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_iin_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_PIN_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_PIN_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_pin_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_VOUT_MIN: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_vout_min);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_VOUT_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_vout_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_IOUT_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_iout_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_POUT_MAX: /* Read-Only word */
- if (pmdev->pages[index].page_flags & PB_HAS_POUT_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_pout_max);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_MAX_TEMP_1: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMP_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_max_temp_1);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_MAX_TEMP_2: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMP_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_max_temp_2);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_MFR_MAX_TEMP_3: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMP_RATING) {
- pmbus_send16(pmdev, pmdev->pages[index].mfr_max_temp_3);
- } else {
- goto passthough;
- }
- break;
- case PMBUS_IDLE_STATE:
- pmbus_send8(pmdev, PMBUS_ERR_BYTE);
- break;
- case PMBUS_CLEAR_FAULTS: /* Send Byte */
- case PMBUS_PAGE_PLUS_WRITE: /* Block Write-only */
- case PMBUS_STORE_DEFAULT_ALL: /* Send Byte */
- case PMBUS_RESTORE_DEFAULT_ALL: /* Send Byte */
- case PMBUS_STORE_DEFAULT_CODE: /* Write-only Byte */
- case PMBUS_RESTORE_DEFAULT_CODE: /* Write-only Byte */
- case PMBUS_STORE_USER_ALL: /* Send Byte */
- case PMBUS_RESTORE_USER_ALL: /* Send Byte */
- case PMBUS_STORE_USER_CODE: /* Write-only Byte */
- case PMBUS_RESTORE_USER_CODE: /* Write-only Byte */
- case PMBUS_QUERY: /* Write-Only */
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: reading from write only register 0x%02x\n",
- __func__, pmdev->code);
- break;
- passthough:
- default:
- /* Pass through read request if not handled */
- if (pmdc->receive_byte) {
- ret = pmdc->receive_byte(pmdev);
- }
- break;
- }
- if (pmdev->out_buf_len != 0) {
- ret = pmbus_out_buf_pop(pmdev);
- return ret;
- }
- return ret;
- }
- /*
- * PMBus clear faults command applies to all status registers, existing faults
- * should separately get re-asserted.
- */
- static void pmbus_clear_faults(PMBusDevice *pmdev)
- {
- for (uint8_t i = 0; i < pmdev->num_pages; i++) {
- pmdev->pages[i].status_word = 0;
- pmdev->pages[i].status_vout = 0;
- pmdev->pages[i].status_iout = 0;
- pmdev->pages[i].status_input = 0;
- pmdev->pages[i].status_temperature = 0;
- pmdev->pages[i].status_cml = 0;
- pmdev->pages[i].status_other = 0;
- pmdev->pages[i].status_mfr_specific = 0;
- pmdev->pages[i].status_fans_1_2 = 0;
- pmdev->pages[i].status_fans_3_4 = 0;
- }
- }
- /*
- * PMBus operation is used to turn On and Off PSUs
- * Therefore, default value for the Operation should be PB_OP_ON or 0x80
- */
- static void pmbus_operation(PMBusDevice *pmdev)
- {
- uint8_t index = pmdev->page;
- if ((pmdev->pages[index].operation & PB_OP_ON) == 0) {
- pmdev->pages[index].read_vout = 0;
- pmdev->pages[index].read_iout = 0;
- pmdev->pages[index].read_pout = 0;
- return;
- }
- if (pmdev->pages[index].operation & (PB_OP_ON | PB_OP_MARGIN_HIGH)) {
- pmdev->pages[index].read_vout = pmdev->pages[index].vout_margin_high;
- }
- if (pmdev->pages[index].operation & (PB_OP_ON | PB_OP_MARGIN_LOW)) {
- pmdev->pages[index].read_vout = pmdev->pages[index].vout_margin_low;
- }
- pmbus_check_limits(pmdev);
- }
- static int pmbus_write_data(SMBusDevice *smd, uint8_t *buf, uint8_t len)
- {
- PMBusDevice *pmdev = PMBUS_DEVICE(smd);
- PMBusDeviceClass *pmdc = PMBUS_DEVICE_GET_CLASS(pmdev);
- int ret = 0;
- uint8_t index;
- if (len == 0) {
- qemu_log_mask(LOG_GUEST_ERROR, "%s: writing empty data\n", __func__);
- return PMBUS_ERR_BYTE;
- }
- if (!pmdev->pages) { /* allocate memory for pages on first use */
- pmbus_pages_alloc(pmdev);
- }
- pmdev->in_buf_len = len;
- pmdev->in_buf = buf;
- pmdev->code = buf[0]; /* PMBus command code */
- if (pmdev->code == PMBUS_CLEAR_FAULTS) {
- pmbus_clear_faults(pmdev);
- }
- if (len == 1) { /* Single length writes are command codes only */
- return 0;
- }
- if (pmdev->code == PMBUS_PAGE) {
- pmdev->page = pmbus_receive8(pmdev);
- if (pmdev->page > pmdev->num_pages - 1 && pmdev->page != PB_ALL_PAGES) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: page %u is out of range\n",
- __func__, pmdev->page);
- pmdev->page = 0; /* undefined behaviour - reset to page 0 */
- pmbus_cml_error(pmdev);
- return PMBUS_ERR_BYTE;
- }
- return 0;
- }
- /* loop through all the pages when 0xFF is received */
- if (pmdev->page == PB_ALL_PAGES) {
- for (int i = 0; i < pmdev->num_pages; i++) {
- pmdev->page = i;
- pmbus_write_data(smd, buf, len);
- }
- pmdev->page = PB_ALL_PAGES;
- return 0;
- }
- index = pmdev->page;
- switch (pmdev->code) {
- case PMBUS_OPERATION: /* R/W byte */
- pmdev->pages[index].operation = pmbus_receive8(pmdev);
- pmbus_operation(pmdev);
- break;
- case PMBUS_ON_OFF_CONFIG: /* R/W byte */
- pmdev->pages[index].on_off_config = pmbus_receive8(pmdev);
- break;
- case PMBUS_CLEAR_FAULTS: /* Send Byte */
- pmbus_clear_faults(pmdev);
- break;
- case PMBUS_PHASE: /* R/W byte */
- pmdev->pages[index].phase = pmbus_receive8(pmdev);
- break;
- case PMBUS_PAGE_PLUS_WRITE: /* Block Write-only */
- case PMBUS_WRITE_PROTECT: /* R/W byte */
- pmdev->pages[index].write_protect = pmbus_receive8(pmdev);
- break;
- case PMBUS_VOUT_MODE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MODE) {
- pmdev->pages[index].vout_mode = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_COMMAND: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_command = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_TRIM: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_trim = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_CAL_OFFSET: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_cal_offset = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_MAX: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_max = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_MARGIN_HIGH: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MARGIN) {
- pmdev->pages[index].vout_margin_high = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_MARGIN_LOW: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_MARGIN) {
- pmdev->pages[index].vout_margin_low = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_TRANSITION_RATE: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_transition_rate = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_DROOP: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_droop = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_SCALE_LOOP: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_scale_loop = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_SCALE_MONITOR: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_scale_monitor = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_MIN: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT_RATING) {
- pmdev->pages[index].vout_min = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_POUT_MAX: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].pout_max = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_ON: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_on = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_OFF: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_off = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_CAL_GAIN: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT_GAIN) {
- pmdev->pages[index].iout_cal_gain = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_CONFIG_1_2: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_config_1_2 = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_COMMAND_1: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_command_1 = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_COMMAND_2: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_command_2 = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_CONFIG_3_4: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_config_3_4 = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_COMMAND_3: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_command_3 = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_FAN_COMMAND_4: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmdev->pages[index].fan_command_4 = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_OV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_ov_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_OV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_ov_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_OV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_ov_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_UV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_uv_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_UV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_uv_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VOUT_UV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].vout_uv_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_OC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_oc_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_OC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_oc_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_OC_LV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_oc_lv_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_OC_LV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_oc_lv_fault_response
- = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_OC_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_oc_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_UC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_uc_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IOUT_UC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].iout_uc_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_OT_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ot_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_OT_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ot_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_OT_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ot_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_UT_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ut_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_UT_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ut_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_UT_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].ut_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_OV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_ov_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_OV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_ov_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_OV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_ov_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_UV_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_uv_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_UV_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_uv_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_VIN_UV_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VIN) {
- pmdev->pages[index].vin_uv_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IIN_OC_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmdev->pages[index].iin_oc_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IIN_OC_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmdev->pages[index].iin_oc_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_IIN_OC_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_IIN) {
- pmdev->pages[index].iin_oc_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_POUT_OP_FAULT_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].pout_op_fault_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_POUT_OP_FAULT_RESPONSE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].pout_op_fault_response = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_POUT_OP_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].pout_op_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_PIN_OP_WARN_LIMIT: /* R/W word */
- if (pmdev->pages[index].page_flags & PB_HAS_PIN) {
- pmdev->pages[index].pin_op_warn_limit = pmbus_receive16(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_STATUS_BYTE: /* R/W byte */
- pmdev->pages[index].status_word = pmbus_receive8(pmdev);
- break;
- case PMBUS_STATUS_WORD: /* R/W word */
- pmdev->pages[index].status_word = pmbus_receive16(pmdev);
- break;
- case PMBUS_STATUS_VOUT: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_VOUT) {
- pmdev->pages[index].status_vout = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_STATUS_IOUT: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_IOUT) {
- pmdev->pages[index].status_iout = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_STATUS_INPUT: /* R/W byte */
- pmdev->pages[index].status_input = pmbus_receive8(pmdev);
- break;
- case PMBUS_STATUS_TEMPERATURE: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_TEMPERATURE) {
- pmdev->pages[index].status_temperature = pmbus_receive8(pmdev);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_STATUS_CML: /* R/W byte */
- pmdev->pages[index].status_cml = pmbus_receive8(pmdev);
- break;
- case PMBUS_STATUS_OTHER: /* R/W byte */
- pmdev->pages[index].status_other = pmbus_receive8(pmdev);
- break;
- case PMBUS_STATUS_MFR_SPECIFIC: /* R/W byte */
- pmdev->pages[index].status_mfr_specific = pmbus_receive8(pmdev);
- break;
- case PMBUS_STATUS_FANS_1_2: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].status_fans_1_2);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_STATUS_FANS_3_4: /* R/W byte */
- if (pmdev->pages[index].page_flags & PB_HAS_FAN) {
- pmbus_send8(pmdev, pmdev->pages[index].status_fans_3_4);
- } else {
- goto passthrough;
- }
- break;
- case PMBUS_PAGE_PLUS_READ: /* Block Read-only */
- case PMBUS_CAPABILITY: /* Read-Only byte */
- case PMBUS_COEFFICIENTS: /* Read-only block 5 bytes */
- case PMBUS_READ_EIN: /* Read-Only block 5 bytes */
- case PMBUS_READ_EOUT: /* Read-Only block 5 bytes */
- case PMBUS_READ_VIN: /* Read-Only word */
- case PMBUS_READ_IIN: /* Read-Only word */
- case PMBUS_READ_VCAP: /* Read-Only word */
- case PMBUS_READ_VOUT: /* Read-Only word */
- case PMBUS_READ_IOUT: /* Read-Only word */
- case PMBUS_READ_TEMPERATURE_1: /* Read-Only word */
- case PMBUS_READ_TEMPERATURE_2: /* Read-Only word */
- case PMBUS_READ_TEMPERATURE_3: /* Read-Only word */
- case PMBUS_READ_FAN_SPEED_1: /* Read-Only word */
- case PMBUS_READ_FAN_SPEED_2: /* Read-Only word */
- case PMBUS_READ_FAN_SPEED_3: /* Read-Only word */
- case PMBUS_READ_FAN_SPEED_4: /* Read-Only word */
- case PMBUS_READ_DUTY_CYCLE: /* Read-Only word */
- case PMBUS_READ_FREQUENCY: /* Read-Only word */
- case PMBUS_READ_POUT: /* Read-Only word */
- case PMBUS_READ_PIN: /* Read-Only word */
- case PMBUS_REVISION: /* Read-Only byte */
- case PMBUS_APP_PROFILE_SUPPORT: /* Read-Only block-read */
- case PMBUS_MFR_VIN_MIN: /* Read-Only word */
- case PMBUS_MFR_VIN_MAX: /* Read-Only word */
- case PMBUS_MFR_IIN_MAX: /* Read-Only word */
- case PMBUS_MFR_PIN_MAX: /* Read-Only word */
- case PMBUS_MFR_VOUT_MIN: /* Read-Only word */
- case PMBUS_MFR_VOUT_MAX: /* Read-Only word */
- case PMBUS_MFR_IOUT_MAX: /* Read-Only word */
- case PMBUS_MFR_POUT_MAX: /* Read-Only word */
- case PMBUS_MFR_TAMBIENT_MAX: /* Read-Only word */
- case PMBUS_MFR_TAMBIENT_MIN: /* Read-Only word */
- case PMBUS_MFR_EFFICIENCY_LL: /* Read-Only block 14 bytes */
- case PMBUS_MFR_EFFICIENCY_HL: /* Read-Only block 14 bytes */
- case PMBUS_MFR_PIN_ACCURACY: /* Read-Only byte */
- case PMBUS_IC_DEVICE_ID: /* Read-Only block-read */
- case PMBUS_IC_DEVICE_REV: /* Read-Only block-read */
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: writing to read-only register 0x%02x\n",
- __func__, pmdev->code);
- break;
- passthrough:
- /* Unimplemented registers get passed to the device */
- default:
- if (pmdc->write_data) {
- ret = pmdc->write_data(pmdev, buf, len);
- }
- break;
- }
- pmbus_check_limits(pmdev);
- pmdev->in_buf_len = 0;
- return ret;
- }
- int pmbus_page_config(PMBusDevice *pmdev, uint8_t index, uint64_t flags)
- {
- if (!pmdev->pages) { /* allocate memory for pages on first use */
- pmbus_pages_alloc(pmdev);
- }
- /* The 0xFF page is special for commands applying to all pages */
- if (index == PB_ALL_PAGES) {
- for (int i = 0; i < pmdev->num_pages; i++) {
- pmdev->pages[i].page_flags = flags;
- }
- return 0;
- }
- if (index > pmdev->num_pages - 1) {
- qemu_log_mask(LOG_GUEST_ERROR,
- "%s: index %u is out of range\n",
- __func__, index);
- return -1;
- }
- pmdev->pages[index].page_flags = flags;
- return 0;
- }
- /* TODO: include pmbus page info in vmstate */
- const VMStateDescription vmstate_pmbus_device = {
- .name = TYPE_PMBUS_DEVICE,
- .version_id = 0,
- .minimum_version_id = 0,
- .fields = (const VMStateField[]) {
- VMSTATE_SMBUS_DEVICE(smb, PMBusDevice),
- VMSTATE_UINT8(num_pages, PMBusDevice),
- VMSTATE_UINT8(code, PMBusDevice),
- VMSTATE_UINT8(page, PMBusDevice),
- VMSTATE_UINT8(capability, PMBusDevice),
- VMSTATE_END_OF_LIST()
- }
- };
- static void pmbus_device_finalize(Object *obj)
- {
- PMBusDevice *pmdev = PMBUS_DEVICE(obj);
- g_free(pmdev->pages);
- }
- static void pmbus_device_class_init(ObjectClass *klass, void *data)
- {
- SMBusDeviceClass *k = SMBUS_DEVICE_CLASS(klass);
- k->quick_cmd = pmbus_quick_cmd;
- k->write_data = pmbus_write_data;
- k->receive_byte = pmbus_receive_byte;
- }
- static const TypeInfo pmbus_device_type_info = {
- .name = TYPE_PMBUS_DEVICE,
- .parent = TYPE_SMBUS_DEVICE,
- .instance_size = sizeof(PMBusDevice),
- .instance_finalize = pmbus_device_finalize,
- .abstract = true,
- .class_size = sizeof(PMBusDeviceClass),
- .class_init = pmbus_device_class_init,
- };
- static void pmbus_device_register_types(void)
- {
- type_register_static(&pmbus_device_type_info);
- }
- type_init(pmbus_device_register_types)
|