root/drivers/media/pci/mgb4/mgb4_i2c.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (C) 2021-2023 Digiteq Automotive
 *     author: Martin Tuma <martin.tuma@digiteqautomotive.com>
 *
 * The i2c module unifies the I2C access to the serializes/deserializes. The I2C
 * chips on the GMSL module use 16b addressing, the FPDL3 chips use standard
 * 8b addressing.
 */

#include "mgb4_i2c.h"

static int read_r16(struct i2c_client *client, u16 reg, u8 *val, int len)
{
        int ret;
        u8 buf[2];
        struct i2c_msg msg[2] = {
                {
                        .addr = client->addr,
                        .flags = 0,
                        .len = 2,
                        .buf = buf,
                }, {
                        .addr = client->addr,
                        .flags = I2C_M_RD,
                        .len = len,
                        .buf = val,
                }
        };

        buf[0] = (reg >> 8) & 0xff;
        buf[1] = (reg >> 0) & 0xff;

        ret = i2c_transfer(client->adapter, msg, 2);
        if (ret < 0)
                return ret;
        else if (ret != 2)
                return -EREMOTEIO;
        else
                return 0;
}

static int write_r16(struct i2c_client *client, u16 reg, const u8 *val, int len)
{
        int ret;
        u8 buf[4];
        struct i2c_msg msg[1] = {
                {
                        .addr = client->addr,
                        .flags = 0,
                        .len = 2 + len,
                        .buf = buf,
                }
        };

        if (2 + len > sizeof(buf))
                return -EINVAL;

        buf[0] = (reg >> 8) & 0xff;
        buf[1] = (reg >> 0) & 0xff;
        memcpy(&buf[2], val, len);

        ret = i2c_transfer(client->adapter, msg, 1);
        if (ret < 0)
                return ret;
        else if (ret != 1)
                return -EREMOTEIO;
        else
                return 0;
}

int mgb4_i2c_init(struct mgb4_i2c_client *client, struct i2c_adapter *adap,
                  struct i2c_board_info const *info, int addr_size)
{
        client->client = i2c_new_client_device(adap, info);
        if (IS_ERR(client->client))
                return PTR_ERR(client->client);

        client->addr_size = addr_size;

        return 0;
}

void mgb4_i2c_free(struct mgb4_i2c_client *client)
{
        i2c_unregister_device(client->client);
}

s32 mgb4_i2c_read_byte(struct mgb4_i2c_client *client, u16 reg)
{
        int ret;
        u8 b;

        if (client->addr_size == 8)
                return i2c_smbus_read_byte_data(client->client, reg);

        ret = read_r16(client->client, reg, &b, 1);
        if (ret < 0)
                return ret;

        return (s32)b;
}

s32 mgb4_i2c_write_byte(struct mgb4_i2c_client *client, u16 reg, u8 val)
{
        if (client->addr_size == 8)
                return i2c_smbus_write_byte_data(client->client, reg, val);
        else
                return write_r16(client->client, reg, &val, 1);
}

s32 mgb4_i2c_mask_byte(struct mgb4_i2c_client *client, u16 reg, u8 mask, u8 val)
{
        s32 ret;

        if (mask != 0xFF) {
                ret = mgb4_i2c_read_byte(client, reg);
                if (ret < 0)
                        return ret;
                val |= (u8)ret & ~mask;
        }

        return mgb4_i2c_write_byte(client, reg, val);
}

int mgb4_i2c_configure(struct mgb4_i2c_client *client,
                       const struct mgb4_i2c_kv *values, size_t count)
{
        size_t i;
        s32 res;

        for (i = 0; i < count; i++) {
                res = mgb4_i2c_mask_byte(client, values[i].reg, values[i].mask,
                                         values[i].val);
                if (res < 0)
                        return res;
        }

        return 0;
}