212 lines
5.2 KiB
C
212 lines
5.2 KiB
C
/* Copyright 2013 The Chromium OS Authors. All rights reserved.
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*
|
|
* Common functions for battery charging.
|
|
*/
|
|
|
|
#include "battery_smart.h"
|
|
#include "charge_state_v2.h"
|
|
#include "charger.h"
|
|
#include "common.h"
|
|
#include "console.h"
|
|
#include "dptf.h"
|
|
#include "host_command.h"
|
|
#include "printf.h"
|
|
#include "util.h"
|
|
#include "hooks.h"
|
|
|
|
/* Console output macros */
|
|
#define CPUTS(outstr) cputs(CC_CHARGER, outstr)
|
|
#define CPRINTS(format, args...) cprints(CC_CHARGER, format, ## args)
|
|
|
|
/* DPTF current limit, -1 = none */
|
|
static int dptf_limit_ma = -1;
|
|
|
|
void dptf_set_charging_current_limit(int ma)
|
|
{
|
|
dptf_limit_ma = ma >= 0 ? ma : -1;
|
|
}
|
|
|
|
int dptf_get_charging_current_limit(void)
|
|
{
|
|
return dptf_limit_ma;
|
|
}
|
|
|
|
static void dptf_disable_hook(void)
|
|
{
|
|
/* Before get to Sx, EC should take control of charger from DPTF */
|
|
dptf_limit_ma = -1;
|
|
}
|
|
DECLARE_HOOK(HOOK_CHIPSET_SUSPEND, dptf_disable_hook, HOOK_PRIO_DEFAULT);
|
|
DECLARE_HOOK(HOOK_CHIPSET_SHUTDOWN, dptf_disable_hook, HOOK_PRIO_DEFAULT);
|
|
|
|
int charger_closest_voltage(int voltage)
|
|
{
|
|
const struct charger_info *info = charger_get_info();
|
|
|
|
/*
|
|
* If the requested voltage is non-zero but below our minimum,
|
|
* return the minimum. See crosbug.com/p/8662.
|
|
*/
|
|
if (voltage > 0 && voltage < info->voltage_min)
|
|
return info->voltage_min;
|
|
|
|
/* Clip to max */
|
|
if (voltage > info->voltage_max)
|
|
return info->voltage_max;
|
|
|
|
/* Otherwise round down to nearest voltage step */
|
|
return voltage - (voltage % info->voltage_step);
|
|
}
|
|
|
|
int charger_closest_current(int current)
|
|
{
|
|
const struct charger_info * const info = charger_get_info();
|
|
|
|
/* Apply DPTF limit if necessary */
|
|
if (dptf_limit_ma >= 0 && current > dptf_limit_ma)
|
|
current = dptf_limit_ma;
|
|
|
|
/*
|
|
* If the requested current is non-zero but below our minimum,
|
|
* return the minimum. See crosbug.com/p/8662.
|
|
*/
|
|
if (current > 0 && current < info->current_min)
|
|
return info->current_min;
|
|
|
|
/* Clip to max */
|
|
if (current > info->current_max)
|
|
return info->current_max;
|
|
|
|
/* Otherwise round down to nearest current step */
|
|
return current - (current % info->current_step);
|
|
}
|
|
|
|
void charger_get_params(struct charger_params *chg)
|
|
{
|
|
memset(chg, 0, sizeof(*chg));
|
|
|
|
if (charger_get_current(&chg->current))
|
|
chg->flags |= CHG_FLAG_BAD_CURRENT;
|
|
|
|
if (charger_get_voltage(&chg->voltage))
|
|
chg->flags |= CHG_FLAG_BAD_VOLTAGE;
|
|
|
|
if (charger_get_input_current(&chg->input_current))
|
|
chg->flags |= CHG_FLAG_BAD_INPUT_CURRENT;
|
|
|
|
if (charger_get_status(&chg->status))
|
|
chg->flags |= CHG_FLAG_BAD_STATUS;
|
|
|
|
if (charger_get_option(&chg->option))
|
|
chg->flags |= CHG_FLAG_BAD_OPTION;
|
|
}
|
|
|
|
static void print_item_name(const char *name)
|
|
{
|
|
ccprintf(" %-8s", name);
|
|
}
|
|
|
|
static int check_print_error(int rv)
|
|
{
|
|
if (rv == EC_SUCCESS)
|
|
return 1;
|
|
ccputs(rv == EC_ERROR_UNIMPLEMENTED ? "(unsupported)\n" : "(error)\n");
|
|
return 0;
|
|
}
|
|
|
|
void print_charger_debug(void)
|
|
{
|
|
int d;
|
|
const struct charger_info *info = charger_get_info();
|
|
|
|
/* info */
|
|
print_item_name("Name:");
|
|
ccprintf("%s\n", info->name);
|
|
|
|
/* option */
|
|
print_item_name("Option:");
|
|
if (check_print_error(charger_get_option(&d)))
|
|
ccprintf("%016b (0x%04x)\n", d, d);
|
|
|
|
/* manufacturer id */
|
|
print_item_name("Man id:");
|
|
if (check_print_error(charger_manufacturer_id(&d)))
|
|
ccprintf("0x%04x\n", d);
|
|
|
|
/* device id */
|
|
print_item_name("Dev id:");
|
|
if (check_print_error(charger_device_id(&d)))
|
|
ccprintf("0x%04x\n", d);
|
|
|
|
/* charge voltage limit */
|
|
print_item_name("V_batt:");
|
|
if (check_print_error(charger_get_voltage(&d)))
|
|
ccprintf("%5d (%4d - %5d, %3d)\n", d,
|
|
info->voltage_min, info->voltage_max,
|
|
info->voltage_step);
|
|
|
|
/* charge current limit */
|
|
print_item_name("I_batt:");
|
|
if (check_print_error(charger_get_current(&d)))
|
|
ccprintf("%5d (%4d - %5d, %3d)\n", d,
|
|
info->current_min, info->current_max,
|
|
info->current_step);
|
|
|
|
/* input current limit */
|
|
print_item_name("I_in:");
|
|
if (check_print_error(charger_get_input_current(&d)))
|
|
ccprintf("%5d (%4d - %5d, %3d)\n", d,
|
|
info->input_current_min, info->input_current_max,
|
|
info->input_current_step);
|
|
|
|
/* dptf current limit */
|
|
print_item_name("I_dptf:");
|
|
if (dptf_limit_ma >= 0)
|
|
ccprintf("%5d\n", dptf_limit_ma);
|
|
else
|
|
ccputs("disabled\n");
|
|
}
|
|
|
|
static int command_charger(int argc, char **argv)
|
|
{
|
|
int d;
|
|
char *e;
|
|
|
|
if (argc != 3) {
|
|
print_charger_debug();
|
|
return EC_SUCCESS;
|
|
}
|
|
|
|
if (strcasecmp(argv[1], "input") == 0) {
|
|
d = strtoi(argv[2], &e, 0);
|
|
if (*e)
|
|
return EC_ERROR_PARAM2;
|
|
return charger_set_input_current(d);
|
|
} else if (strcasecmp(argv[1], "current") == 0) {
|
|
d = strtoi(argv[2], &e, 0);
|
|
if (*e)
|
|
return EC_ERROR_PARAM2;
|
|
chgstate_set_manual_current(d);
|
|
return charger_set_current(d);
|
|
} else if (strcasecmp(argv[1], "voltage") == 0) {
|
|
d = strtoi(argv[2], &e, 0);
|
|
if (*e)
|
|
return EC_ERROR_PARAM2;
|
|
chgstate_set_manual_voltage(d);
|
|
return charger_set_voltage(d);
|
|
} else if (strcasecmp(argv[1], "dptf") == 0) {
|
|
d = strtoi(argv[2], &e, 0);
|
|
if (*e)
|
|
return EC_ERROR_PARAM2;
|
|
dptf_limit_ma = d;
|
|
return EC_SUCCESS;
|
|
} else
|
|
return EC_ERROR_PARAM1;
|
|
}
|
|
|
|
DECLARE_CONSOLE_COMMAND(charger, command_charger,
|
|
"[input | current | voltage | dptf] [newval]",
|
|
"Get or set charger param(s)");
|