ectool: query packet size and set them properly internally.

Allow to negotiate the packet command and responses to
whatever values the EC can support.
Set the buffer size including the necessary V3 header.

If the EC run v3 protocol with large packet support, but the kernel
does not have v3 support (3.10), we can not support sending or
receiving large commands.
Be aware that on 3.10, commands like ectool console will fail if
the EC wants to send command larger than the kernel can handle.

Copied kernel_version_ge from libusb-1.0.19/libusb/os/linux_usbfs.c.

BUG=chrome-os-partner:31989,chrome-os-partner:31660,chromium:454324,chrome-os-partner:39265
BRANCH=none
TEST=Build a special firmware to exchange 300 bytes.
Check ectool console works with the righ size.
Check that ectool is calling uname.
Check on Nyan_big: without change, "ectool version" crashes kernel. With
changes, "ectool version" works.

In conseuqence, it reverts commit be0bd9b835,
"ectool: Do not increase buffer size after probe max size from ec"

Change-Id: I54ffd43488ea81272f30789dc87a261085769fe0
Signed-off-by: Gwendal Grignou <gwendal@chromium.org>
Reviewed-on: https://chromium-review.googlesource.com/274086
Reviewed-by: Shawn N <shawnn@chromium.org>
This commit is contained in:
Gwendal Grignou
2015-03-24 17:11:09 -07:00
committed by ChromeOS Commit Bot
parent 1ade8e02a7
commit c89ebdb66c
4 changed files with 59 additions and 6 deletions

View File

@@ -10,9 +10,9 @@ host-util-bin=ectool lbplay stm32mon ec_sb_firmware_update lbcc
build-util-bin=ec_uartd iteflash
comm-objs=$(util-lock-objs:%=lock/%) comm-host.o comm-dev.o
comm-objs+=comm-lpc.o comm-i2c.o
comm-objs+=comm-lpc.o comm-i2c.o misc_util.o
ectool-objs=ectool.o ectool_keyscan.o misc_util.o ec_flash.o $(comm-objs)
ectool-objs=ectool.o ectool_keyscan.o ec_flash.o $(comm-objs)
ec_sb_firmware_update-objs=ec_sb_firmware_update.o $(comm-objs) misc_util.o
ec_sb_firmware_update-objs+=powerd_lock.o
lbplay-objs=lbplay.o $(comm-objs)

View File

@@ -11,6 +11,8 @@
#include "comm-host.h"
#include "ec_commands.h"
#include "misc_util.h"
int (*ec_command_proto)(int command, int version,
const void *outdata, int outsize,
@@ -78,10 +80,17 @@ int ec_command(int command, int version,
int comm_init(int interfaces, const char *device_name)
{
struct ec_response_get_protocol_info info;
int allow_large_buffer;
/* Default memmap access */
ec_readmem = fake_readmem;
allow_large_buffer = kernel_version_ge(3, 14, 0);
if (allow_large_buffer < 0) {
fprintf(stderr, "Unable to check linux version\n");
return 1;
}
/* Prefer new /dev method */
if ((interfaces & COMM_DEV) && comm_init_dev &&
!comm_init_dev(device_name))
@@ -111,10 +120,12 @@ int comm_init(int interfaces, const char *device_name)
/* read max request / response size from ec for protocol v3+ */
if (ec_command(EC_CMD_GET_PROTOCOL_INFO, 0, NULL, 0, &info,
sizeof(info)) == sizeof(info)) {
ec_max_outsize = info.max_request_packet_size -
sizeof(struct ec_host_request);
ec_max_insize = info.max_response_packet_size -
sizeof(struct ec_host_response);
if ((allow_large_buffer) ||
(info.max_request_packet_size < ec_max_outsize))
ec_max_outsize = info.max_request_packet_size;
if ((allow_large_buffer) ||
(info.max_request_packet_size < ec_max_insize))
ec_max_insize = info.max_response_packet_size;
ec_outbuf = realloc(ec_outbuf, ec_max_outsize);
ec_inbuf = realloc(ec_inbuf, ec_max_insize);

View File

@@ -8,6 +8,7 @@
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/utsname.h>
#include "comm-host.h"
#include "misc_util.h"
@@ -133,3 +134,39 @@ int ec_cmd_version_supported(int cmd, int ver)
return (mask & EC_VER_MASK(ver)) ? 1 : 0;
}
/**
* Return 1 is the current kernel version is greater or equal to
* <major>.<minor>.<sublevel>
*/
int kernel_version_ge(int major, int minor, int sublevel)
{
struct utsname uts;
int atoms, kmajor, kminor, ksublevel;
if (uname(&uts) < 0)
return -1;
atoms = sscanf(uts.release, "%d.%d.%d", &kmajor, &kminor, &ksublevel);
if (atoms < 1)
return -1;
if (kmajor > major)
return 1;
if (kmajor < major)
return 0;
/* kmajor == major */
if (atoms < 2)
return 0 == minor && 0 == sublevel;
if (kminor > minor)
return 1;
if (kminor < minor)
return 0;
/* kminor == minor */
if (atoms < 3)
return 0 == sublevel;
return ksublevel >= sublevel;
}

View File

@@ -56,4 +56,9 @@ int ec_get_cmd_versions(int cmd, uint32_t *pmask);
*/
int ec_cmd_version_supported(int cmd, int ver);
/**
* Return 1 is the current kernel version is greater or equal to
* <major>.<minor>.<sublevel>
*/
int kernel_version_ge(int major, int minor, int sublevel);
#endif