mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-07 14:24:18 +08:00
e9a6f5bca5
Add MMSYS support for the MT6795 SoC using the same mmsys routing table as MT8173 as, for the currently supported usecases (DSI0, DPI0 with no WDMA), these are identical. Signed-off-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com> Link: https://lore.kernel.org/r/20230309102618.114157-4-angelogioacchino.delregno@collabora.com Signed-off-by: Matthias Brugger <matthias.bgg@gmail.com>
459 lines
14 KiB
C
459 lines
14 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (c) 2014 MediaTek Inc.
|
|
* Author: James Liao <jamesjj.liao@mediatek.com>
|
|
*/
|
|
|
|
#include <linux/delay.h>
|
|
#include <linux/device.h>
|
|
#include <linux/io.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of_device.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/reset-controller.h>
|
|
#include <linux/soc/mediatek/mtk-mmsys.h>
|
|
|
|
#include "mtk-mmsys.h"
|
|
#include "mt8167-mmsys.h"
|
|
#include "mt8173-mmsys.h"
|
|
#include "mt8183-mmsys.h"
|
|
#include "mt8186-mmsys.h"
|
|
#include "mt8188-mmsys.h"
|
|
#include "mt8192-mmsys.h"
|
|
#include "mt8195-mmsys.h"
|
|
#include "mt8365-mmsys.h"
|
|
|
|
#define MMSYS_SW_RESET_PER_REG 32
|
|
|
|
static const struct mtk_mmsys_driver_data mt2701_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt2701-mm",
|
|
.routes = mmsys_default_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_default_routing_table),
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt2712_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt2712-mm",
|
|
.routes = mmsys_default_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_default_routing_table),
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt6779_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt6779-mm",
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt6795_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt6795-mm",
|
|
.routes = mt8173_mmsys_routing_table,
|
|
.num_routes = ARRAY_SIZE(mt8173_mmsys_routing_table),
|
|
.sw0_rst_offset = MT8183_MMSYS_SW0_RST_B,
|
|
.num_resets = 64,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt6797_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt6797-mm",
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8167_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8167-mm",
|
|
.routes = mt8167_mmsys_routing_table,
|
|
.num_routes = ARRAY_SIZE(mt8167_mmsys_routing_table),
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8173_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8173-mm",
|
|
.routes = mt8173_mmsys_routing_table,
|
|
.num_routes = ARRAY_SIZE(mt8173_mmsys_routing_table),
|
|
.sw0_rst_offset = MT8183_MMSYS_SW0_RST_B,
|
|
.num_resets = 64,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8183_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8183-mm",
|
|
.routes = mmsys_mt8183_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8183_routing_table),
|
|
.sw0_rst_offset = MT8183_MMSYS_SW0_RST_B,
|
|
.num_resets = 32,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8186_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8186-mm",
|
|
.routes = mmsys_mt8186_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8186_routing_table),
|
|
.sw0_rst_offset = MT8186_MMSYS_SW0_RST_B,
|
|
.num_resets = 32,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8188_vdosys0_driver_data = {
|
|
.clk_driver = "clk-mt8188-vdo0",
|
|
.routes = mmsys_mt8188_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8188_routing_table),
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8192_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8192-mm",
|
|
.routes = mmsys_mt8192_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8192_routing_table),
|
|
.sw0_rst_offset = MT8186_MMSYS_SW0_RST_B,
|
|
.num_resets = 32,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8195_vdosys0_driver_data = {
|
|
.clk_driver = "clk-mt8195-vdo0",
|
|
.routes = mmsys_mt8195_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8195_routing_table),
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8195_vdosys1_driver_data = {
|
|
.clk_driver = "clk-mt8195-vdo1",
|
|
.routes = mmsys_mt8195_vdo1_routing_table,
|
|
.num_routes = ARRAY_SIZE(mmsys_mt8195_vdo1_routing_table),
|
|
.sw0_rst_offset = MT8195_VDO1_SW0_RST_B,
|
|
.num_resets = 64,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8195_vppsys0_driver_data = {
|
|
.clk_driver = "clk-mt8195-vpp0",
|
|
.is_vppsys = true,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8195_vppsys1_driver_data = {
|
|
.clk_driver = "clk-mt8195-vpp1",
|
|
.is_vppsys = true,
|
|
};
|
|
|
|
static const struct mtk_mmsys_driver_data mt8365_mmsys_driver_data = {
|
|
.clk_driver = "clk-mt8365-mm",
|
|
.routes = mt8365_mmsys_routing_table,
|
|
.num_routes = ARRAY_SIZE(mt8365_mmsys_routing_table),
|
|
};
|
|
|
|
struct mtk_mmsys {
|
|
void __iomem *regs;
|
|
const struct mtk_mmsys_driver_data *data;
|
|
struct platform_device *clks_pdev;
|
|
struct platform_device *drm_pdev;
|
|
spinlock_t lock; /* protects mmsys_sw_rst_b reg */
|
|
struct reset_controller_dev rcdev;
|
|
struct cmdq_client_reg cmdq_base;
|
|
};
|
|
|
|
static void mtk_mmsys_update_bits(struct mtk_mmsys *mmsys, u32 offset, u32 mask, u32 val,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
int ret;
|
|
u32 tmp;
|
|
|
|
if (mmsys->cmdq_base.size && cmdq_pkt) {
|
|
ret = cmdq_pkt_write_mask(cmdq_pkt, mmsys->cmdq_base.subsys,
|
|
mmsys->cmdq_base.offset + offset, val,
|
|
mask);
|
|
if (ret)
|
|
pr_debug("CMDQ unavailable: using CPU write\n");
|
|
else
|
|
return;
|
|
}
|
|
tmp = readl_relaxed(mmsys->regs + offset);
|
|
tmp = (tmp & ~mask) | (val & mask);
|
|
writel_relaxed(tmp, mmsys->regs + offset);
|
|
}
|
|
|
|
void mtk_mmsys_ddp_connect(struct device *dev,
|
|
enum mtk_ddp_comp_id cur,
|
|
enum mtk_ddp_comp_id next)
|
|
{
|
|
struct mtk_mmsys *mmsys = dev_get_drvdata(dev);
|
|
const struct mtk_mmsys_routes *routes = mmsys->data->routes;
|
|
int i;
|
|
|
|
for (i = 0; i < mmsys->data->num_routes; i++)
|
|
if (cur == routes[i].from_comp && next == routes[i].to_comp)
|
|
mtk_mmsys_update_bits(mmsys, routes[i].addr, routes[i].mask,
|
|
routes[i].val, NULL);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_ddp_connect);
|
|
|
|
void mtk_mmsys_ddp_disconnect(struct device *dev,
|
|
enum mtk_ddp_comp_id cur,
|
|
enum mtk_ddp_comp_id next)
|
|
{
|
|
struct mtk_mmsys *mmsys = dev_get_drvdata(dev);
|
|
const struct mtk_mmsys_routes *routes = mmsys->data->routes;
|
|
int i;
|
|
|
|
for (i = 0; i < mmsys->data->num_routes; i++)
|
|
if (cur == routes[i].from_comp && next == routes[i].to_comp)
|
|
mtk_mmsys_update_bits(mmsys, routes[i].addr, routes[i].mask, 0, NULL);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_ddp_disconnect);
|
|
|
|
void mtk_mmsys_merge_async_config(struct device *dev, int idx, int width, int height,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev), MT8195_VDO1_MERGE0_ASYNC_CFG_WD + 0x10 * idx,
|
|
~0, height << 16 | width, cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_merge_async_config);
|
|
|
|
void mtk_mmsys_hdr_config(struct device *dev, int be_width, int be_height,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev), MT8195_VDO1_HDRBE_ASYNC_CFG_WD, ~0,
|
|
be_height << 16 | be_width, cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_hdr_config);
|
|
|
|
void mtk_mmsys_mixer_in_config(struct device *dev, int idx, bool alpha_sel, u16 alpha,
|
|
u8 mode, u32 biwidth, struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
struct mtk_mmsys *mmsys = dev_get_drvdata(dev);
|
|
|
|
mtk_mmsys_update_bits(mmsys, MT8195_VDO1_MIXER_IN1_ALPHA + (idx - 1) * 4, ~0,
|
|
alpha << 16 | alpha, cmdq_pkt);
|
|
mtk_mmsys_update_bits(mmsys, MT8195_VDO1_HDR_TOP_CFG, BIT(19 + idx),
|
|
alpha_sel << (19 + idx), cmdq_pkt);
|
|
mtk_mmsys_update_bits(mmsys, MT8195_VDO1_MIXER_IN1_PAD + (idx - 1) * 4,
|
|
GENMASK(31, 16) | GENMASK(1, 0), biwidth << 16 | mode, cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_mixer_in_config);
|
|
|
|
void mtk_mmsys_mixer_in_channel_swap(struct device *dev, int idx, bool channel_swap,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev), MT8195_VDO1_MIXER_IN1_PAD + (idx - 1) * 4,
|
|
BIT(4), channel_swap << 4, cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_mixer_in_channel_swap);
|
|
|
|
void mtk_mmsys_ddp_dpi_fmt_config(struct device *dev, u32 val)
|
|
{
|
|
struct mtk_mmsys *mmsys = dev_get_drvdata(dev);
|
|
|
|
switch (val) {
|
|
case MTK_DPI_RGB888_SDR_CON:
|
|
mtk_mmsys_update_bits(mmsys, MT8186_MMSYS_DPI_OUTPUT_FORMAT,
|
|
MT8186_DPI_FORMAT_MASK, MT8186_DPI_RGB888_SDR_CON, NULL);
|
|
break;
|
|
case MTK_DPI_RGB565_SDR_CON:
|
|
mtk_mmsys_update_bits(mmsys, MT8186_MMSYS_DPI_OUTPUT_FORMAT,
|
|
MT8186_DPI_FORMAT_MASK, MT8186_DPI_RGB565_SDR_CON, NULL);
|
|
break;
|
|
case MTK_DPI_RGB565_DDR_CON:
|
|
mtk_mmsys_update_bits(mmsys, MT8186_MMSYS_DPI_OUTPUT_FORMAT,
|
|
MT8186_DPI_FORMAT_MASK, MT8186_DPI_RGB565_DDR_CON, NULL);
|
|
break;
|
|
case MTK_DPI_RGB888_DDR_CON:
|
|
default:
|
|
mtk_mmsys_update_bits(mmsys, MT8186_MMSYS_DPI_OUTPUT_FORMAT,
|
|
MT8186_DPI_FORMAT_MASK, MT8186_DPI_RGB888_DDR_CON, NULL);
|
|
break;
|
|
}
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_ddp_dpi_fmt_config);
|
|
|
|
void mtk_mmsys_vpp_rsz_merge_config(struct device *dev, u32 id, bool enable,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
u32 reg;
|
|
|
|
switch (id) {
|
|
case 2:
|
|
reg = MT8195_SVPP2_BUF_BF_RSZ_SWITCH;
|
|
break;
|
|
case 3:
|
|
reg = MT8195_SVPP3_BUF_BF_RSZ_SWITCH;
|
|
break;
|
|
default:
|
|
dev_err(dev, "Invalid id %d\n", id);
|
|
return;
|
|
}
|
|
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev), reg, ~0, enable, cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_vpp_rsz_merge_config);
|
|
|
|
void mtk_mmsys_vpp_rsz_dcm_config(struct device *dev, bool enable,
|
|
struct cmdq_pkt *cmdq_pkt)
|
|
{
|
|
u32 client;
|
|
|
|
client = MT8195_SVPP1_MDP_RSZ;
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev),
|
|
MT8195_VPP1_HW_DCM_1ST_DIS0, client,
|
|
((enable) ? client : 0), cmdq_pkt);
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev),
|
|
MT8195_VPP1_HW_DCM_2ND_DIS0, client,
|
|
((enable) ? client : 0), cmdq_pkt);
|
|
|
|
client = MT8195_SVPP2_MDP_RSZ | MT8195_SVPP3_MDP_RSZ;
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev),
|
|
MT8195_VPP1_HW_DCM_1ST_DIS1, client,
|
|
((enable) ? client : 0), cmdq_pkt);
|
|
mtk_mmsys_update_bits(dev_get_drvdata(dev),
|
|
MT8195_VPP1_HW_DCM_2ND_DIS1, client,
|
|
((enable) ? client : 0), cmdq_pkt);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_mmsys_vpp_rsz_dcm_config);
|
|
|
|
static int mtk_mmsys_reset_update(struct reset_controller_dev *rcdev, unsigned long id,
|
|
bool assert)
|
|
{
|
|
struct mtk_mmsys *mmsys = container_of(rcdev, struct mtk_mmsys, rcdev);
|
|
unsigned long flags;
|
|
u32 offset;
|
|
u32 reg;
|
|
|
|
offset = (id / MMSYS_SW_RESET_PER_REG) * sizeof(u32);
|
|
id = id % MMSYS_SW_RESET_PER_REG;
|
|
reg = mmsys->data->sw0_rst_offset + offset;
|
|
|
|
spin_lock_irqsave(&mmsys->lock, flags);
|
|
|
|
if (assert)
|
|
mtk_mmsys_update_bits(mmsys, reg, BIT(id), 0, NULL);
|
|
else
|
|
mtk_mmsys_update_bits(mmsys, reg, BIT(id), BIT(id), NULL);
|
|
|
|
spin_unlock_irqrestore(&mmsys->lock, flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mtk_mmsys_reset_assert(struct reset_controller_dev *rcdev, unsigned long id)
|
|
{
|
|
return mtk_mmsys_reset_update(rcdev, id, true);
|
|
}
|
|
|
|
static int mtk_mmsys_reset_deassert(struct reset_controller_dev *rcdev, unsigned long id)
|
|
{
|
|
return mtk_mmsys_reset_update(rcdev, id, false);
|
|
}
|
|
|
|
static int mtk_mmsys_reset(struct reset_controller_dev *rcdev, unsigned long id)
|
|
{
|
|
int ret;
|
|
|
|
ret = mtk_mmsys_reset_assert(rcdev, id);
|
|
if (ret)
|
|
return ret;
|
|
|
|
usleep_range(1000, 1100);
|
|
|
|
return mtk_mmsys_reset_deassert(rcdev, id);
|
|
}
|
|
|
|
static const struct reset_control_ops mtk_mmsys_reset_ops = {
|
|
.assert = mtk_mmsys_reset_assert,
|
|
.deassert = mtk_mmsys_reset_deassert,
|
|
.reset = mtk_mmsys_reset,
|
|
};
|
|
|
|
static int mtk_mmsys_probe(struct platform_device *pdev)
|
|
{
|
|
struct device *dev = &pdev->dev;
|
|
struct platform_device *clks;
|
|
struct platform_device *drm;
|
|
struct mtk_mmsys *mmsys;
|
|
int ret;
|
|
|
|
mmsys = devm_kzalloc(dev, sizeof(*mmsys), GFP_KERNEL);
|
|
if (!mmsys)
|
|
return -ENOMEM;
|
|
|
|
mmsys->regs = devm_platform_ioremap_resource(pdev, 0);
|
|
if (IS_ERR(mmsys->regs)) {
|
|
ret = PTR_ERR(mmsys->regs);
|
|
dev_err(dev, "Failed to ioremap mmsys registers: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
mmsys->data = of_device_get_match_data(&pdev->dev);
|
|
|
|
if (mmsys->data->num_resets > 0) {
|
|
spin_lock_init(&mmsys->lock);
|
|
|
|
mmsys->rcdev.owner = THIS_MODULE;
|
|
mmsys->rcdev.nr_resets = mmsys->data->num_resets;
|
|
mmsys->rcdev.ops = &mtk_mmsys_reset_ops;
|
|
mmsys->rcdev.of_node = pdev->dev.of_node;
|
|
ret = devm_reset_controller_register(&pdev->dev, &mmsys->rcdev);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "Couldn't register mmsys reset controller: %d\n", ret);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
/* CMDQ is optional */
|
|
ret = cmdq_dev_get_client_reg(dev, &mmsys->cmdq_base, 0);
|
|
if (ret)
|
|
dev_dbg(dev, "No mediatek,gce-client-reg!\n");
|
|
|
|
platform_set_drvdata(pdev, mmsys);
|
|
|
|
clks = platform_device_register_data(&pdev->dev, mmsys->data->clk_driver,
|
|
PLATFORM_DEVID_AUTO, NULL, 0);
|
|
if (IS_ERR(clks))
|
|
return PTR_ERR(clks);
|
|
mmsys->clks_pdev = clks;
|
|
|
|
if (mmsys->data->is_vppsys)
|
|
goto out_probe_done;
|
|
|
|
drm = platform_device_register_data(&pdev->dev, "mediatek-drm",
|
|
PLATFORM_DEVID_AUTO, NULL, 0);
|
|
if (IS_ERR(drm)) {
|
|
platform_device_unregister(clks);
|
|
return PTR_ERR(drm);
|
|
}
|
|
mmsys->drm_pdev = drm;
|
|
|
|
out_probe_done:
|
|
return 0;
|
|
}
|
|
|
|
static int mtk_mmsys_remove(struct platform_device *pdev)
|
|
{
|
|
struct mtk_mmsys *mmsys = platform_get_drvdata(pdev);
|
|
|
|
platform_device_unregister(mmsys->drm_pdev);
|
|
platform_device_unregister(mmsys->clks_pdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct of_device_id of_match_mtk_mmsys[] = {
|
|
{ .compatible = "mediatek,mt2701-mmsys", .data = &mt2701_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt2712-mmsys", .data = &mt2712_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt6779-mmsys", .data = &mt6779_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt6795-mmsys", .data = &mt6795_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt6797-mmsys", .data = &mt6797_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt8167-mmsys", .data = &mt8167_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt8173-mmsys", .data = &mt8173_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt8183-mmsys", .data = &mt8183_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt8186-mmsys", .data = &mt8186_mmsys_driver_data },
|
|
{ .compatible = "mediatek,mt8188-vdosys0", .data = &mt8188_vdosys0_driver_data },
|
|
{ .compatible = "mediatek,mt8192-mmsys", .data = &mt8192_mmsys_driver_data },
|
|
/* "mediatek,mt8195-mmsys" compatible is deprecated */
|
|
{ .compatible = "mediatek,mt8195-mmsys", .data = &mt8195_vdosys0_driver_data },
|
|
{ .compatible = "mediatek,mt8195-vdosys0", .data = &mt8195_vdosys0_driver_data },
|
|
{ .compatible = "mediatek,mt8195-vdosys1", .data = &mt8195_vdosys1_driver_data },
|
|
{ .compatible = "mediatek,mt8195-vppsys0", .data = &mt8195_vppsys0_driver_data },
|
|
{ .compatible = "mediatek,mt8195-vppsys1", .data = &mt8195_vppsys1_driver_data },
|
|
{ .compatible = "mediatek,mt8365-mmsys", .data = &mt8365_mmsys_driver_data },
|
|
{ /* sentinel */ }
|
|
};
|
|
MODULE_DEVICE_TABLE(of, of_match_mtk_mmsys);
|
|
|
|
static struct platform_driver mtk_mmsys_drv = {
|
|
.driver = {
|
|
.name = "mtk-mmsys",
|
|
.of_match_table = of_match_mtk_mmsys,
|
|
},
|
|
.probe = mtk_mmsys_probe,
|
|
.remove = mtk_mmsys_remove,
|
|
};
|
|
module_platform_driver(mtk_mmsys_drv);
|
|
|
|
MODULE_AUTHOR("Yongqiang Niu <yongqiang.niu@mediatek.com>");
|
|
MODULE_DESCRIPTION("MediaTek SoC MMSYS driver");
|
|
MODULE_LICENSE("GPL");
|