162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * max9850.c  --  codec driver for max9850
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2011 taskit GmbH
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Author: Christian Glindkamp <christian.glindkamp@taskit.de>
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * Initial development of this code was funded by
1062306a36Sopenharmony_ci * MICRONIC Computer Systeme GmbH, https://www.mcsberlin.de/
1162306a36Sopenharmony_ci */
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_ci#include <linux/module.h>
1462306a36Sopenharmony_ci#include <linux/init.h>
1562306a36Sopenharmony_ci#include <linux/i2c.h>
1662306a36Sopenharmony_ci#include <linux/regmap.h>
1762306a36Sopenharmony_ci#include <linux/slab.h>
1862306a36Sopenharmony_ci#include <sound/pcm.h>
1962306a36Sopenharmony_ci#include <sound/pcm_params.h>
2062306a36Sopenharmony_ci#include <sound/soc.h>
2162306a36Sopenharmony_ci#include <sound/tlv.h>
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci#include "max9850.h"
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_cistruct max9850_priv {
2662306a36Sopenharmony_ci	struct regmap *regmap;
2762306a36Sopenharmony_ci	unsigned int sysclk;
2862306a36Sopenharmony_ci};
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci/* these registers are not used at the moment but provided for the sake of
3162306a36Sopenharmony_ci * completeness */
3262306a36Sopenharmony_cistatic bool max9850_volatile_register(struct device *dev, unsigned int reg)
3362306a36Sopenharmony_ci{
3462306a36Sopenharmony_ci	switch (reg) {
3562306a36Sopenharmony_ci	case MAX9850_STATUSA:
3662306a36Sopenharmony_ci	case MAX9850_STATUSB:
3762306a36Sopenharmony_ci		return true;
3862306a36Sopenharmony_ci	default:
3962306a36Sopenharmony_ci		return false;
4062306a36Sopenharmony_ci	}
4162306a36Sopenharmony_ci}
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_cistatic const struct regmap_config max9850_regmap = {
4462306a36Sopenharmony_ci	.reg_bits = 8,
4562306a36Sopenharmony_ci	.val_bits = 8,
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_ci	.max_register = MAX9850_DIGITAL_AUDIO,
4862306a36Sopenharmony_ci	.volatile_reg = max9850_volatile_register,
4962306a36Sopenharmony_ci	.cache_type = REGCACHE_RBTREE,
5062306a36Sopenharmony_ci};
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_cistatic const DECLARE_TLV_DB_RANGE(max9850_tlv,
5362306a36Sopenharmony_ci	0x18, 0x1f, TLV_DB_SCALE_ITEM(-7450, 400, 0),
5462306a36Sopenharmony_ci	0x20, 0x33, TLV_DB_SCALE_ITEM(-4150, 200, 0),
5562306a36Sopenharmony_ci	0x34, 0x37, TLV_DB_SCALE_ITEM(-150, 100, 0),
5662306a36Sopenharmony_ci	0x38, 0x3f, TLV_DB_SCALE_ITEM(250, 50, 0)
5762306a36Sopenharmony_ci);
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_cistatic const struct snd_kcontrol_new max9850_controls[] = {
6062306a36Sopenharmony_ciSOC_SINGLE_TLV("Headphone Volume", MAX9850_VOLUME, 0, 0x3f, 1, max9850_tlv),
6162306a36Sopenharmony_ciSOC_SINGLE("Headphone Switch", MAX9850_VOLUME, 7, 1, 1),
6262306a36Sopenharmony_ciSOC_SINGLE("Mono Switch", MAX9850_GENERAL_PURPOSE, 2, 1, 0),
6362306a36Sopenharmony_ci};
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_cistatic const struct snd_kcontrol_new max9850_mixer_controls[] = {
6662306a36Sopenharmony_ci	SOC_DAPM_SINGLE("Line In Switch", MAX9850_ENABLE, 1, 1, 0),
6762306a36Sopenharmony_ci};
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_cistatic const struct snd_soc_dapm_widget max9850_dapm_widgets[] = {
7062306a36Sopenharmony_ciSND_SOC_DAPM_SUPPLY("Charge Pump 1", MAX9850_ENABLE, 4, 0, NULL, 0),
7162306a36Sopenharmony_ciSND_SOC_DAPM_SUPPLY("Charge Pump 2", MAX9850_ENABLE, 5, 0, NULL, 0),
7262306a36Sopenharmony_ciSND_SOC_DAPM_SUPPLY("MCLK", MAX9850_ENABLE, 6, 0, NULL, 0),
7362306a36Sopenharmony_ciSND_SOC_DAPM_SUPPLY("SHDN", MAX9850_ENABLE, 7, 0, NULL, 0),
7462306a36Sopenharmony_ciSND_SOC_DAPM_MIXER_NAMED_CTL("Output Mixer", MAX9850_ENABLE, 2, 0,
7562306a36Sopenharmony_ci		&max9850_mixer_controls[0],
7662306a36Sopenharmony_ci		ARRAY_SIZE(max9850_mixer_controls)),
7762306a36Sopenharmony_ciSND_SOC_DAPM_PGA("Headphone Output", MAX9850_ENABLE, 3, 0, NULL, 0),
7862306a36Sopenharmony_ciSND_SOC_DAPM_DAC("DAC", "HiFi Playback", MAX9850_ENABLE, 0, 0),
7962306a36Sopenharmony_ciSND_SOC_DAPM_OUTPUT("OUTL"),
8062306a36Sopenharmony_ciSND_SOC_DAPM_OUTPUT("HPL"),
8162306a36Sopenharmony_ciSND_SOC_DAPM_OUTPUT("OUTR"),
8262306a36Sopenharmony_ciSND_SOC_DAPM_OUTPUT("HPR"),
8362306a36Sopenharmony_ciSND_SOC_DAPM_MIXER("Line Input", SND_SOC_NOPM, 0, 0, NULL, 0),
8462306a36Sopenharmony_ciSND_SOC_DAPM_INPUT("INL"),
8562306a36Sopenharmony_ciSND_SOC_DAPM_INPUT("INR"),
8662306a36Sopenharmony_ci};
8762306a36Sopenharmony_ci
8862306a36Sopenharmony_cistatic const struct snd_soc_dapm_route max9850_dapm_routes[] = {
8962306a36Sopenharmony_ci	/* output mixer */
9062306a36Sopenharmony_ci	{"Output Mixer", NULL, "DAC"},
9162306a36Sopenharmony_ci	{"Output Mixer", "Line In Switch", "Line Input"},
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	/* outputs */
9462306a36Sopenharmony_ci	{"Headphone Output", NULL, "Output Mixer"},
9562306a36Sopenharmony_ci	{"HPL", NULL, "Headphone Output"},
9662306a36Sopenharmony_ci	{"HPR", NULL, "Headphone Output"},
9762306a36Sopenharmony_ci	{"OUTL", NULL, "Output Mixer"},
9862306a36Sopenharmony_ci	{"OUTR", NULL, "Output Mixer"},
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ci	/* inputs */
10162306a36Sopenharmony_ci	{"Line Input", NULL, "INL"},
10262306a36Sopenharmony_ci	{"Line Input", NULL, "INR"},
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	/* supplies */
10562306a36Sopenharmony_ci	{"Output Mixer", NULL, "Charge Pump 1"},
10662306a36Sopenharmony_ci	{"Output Mixer", NULL, "Charge Pump 2"},
10762306a36Sopenharmony_ci	{"Output Mixer", NULL, "SHDN"},
10862306a36Sopenharmony_ci	{"DAC", NULL, "MCLK"},
10962306a36Sopenharmony_ci};
11062306a36Sopenharmony_ci
11162306a36Sopenharmony_cistatic int max9850_hw_params(struct snd_pcm_substream *substream,
11262306a36Sopenharmony_ci			     struct snd_pcm_hw_params *params,
11362306a36Sopenharmony_ci			     struct snd_soc_dai *dai)
11462306a36Sopenharmony_ci{
11562306a36Sopenharmony_ci	struct snd_soc_component *component = dai->component;
11662306a36Sopenharmony_ci	struct max9850_priv *max9850 = snd_soc_component_get_drvdata(component);
11762306a36Sopenharmony_ci	u64 lrclk_div;
11862306a36Sopenharmony_ci	u8 sf, da;
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	if (!max9850->sysclk)
12162306a36Sopenharmony_ci		return -EINVAL;
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci	/* lrclk_div = 2^22 * rate / iclk with iclk = mclk / sf */
12462306a36Sopenharmony_ci	sf = (snd_soc_component_read(component, MAX9850_CLOCK) >> 2) + 1;
12562306a36Sopenharmony_ci	lrclk_div = (1 << 22);
12662306a36Sopenharmony_ci	lrclk_div *= params_rate(params);
12762306a36Sopenharmony_ci	lrclk_div *= sf;
12862306a36Sopenharmony_ci	do_div(lrclk_div, max9850->sysclk);
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci	snd_soc_component_write(component, MAX9850_LRCLK_MSB, (lrclk_div >> 8) & 0x7f);
13162306a36Sopenharmony_ci	snd_soc_component_write(component, MAX9850_LRCLK_LSB, lrclk_div & 0xff);
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci	switch (params_width(params)) {
13462306a36Sopenharmony_ci	case 16:
13562306a36Sopenharmony_ci		da = 0;
13662306a36Sopenharmony_ci		break;
13762306a36Sopenharmony_ci	case 20:
13862306a36Sopenharmony_ci		da = 0x2;
13962306a36Sopenharmony_ci		break;
14062306a36Sopenharmony_ci	case 24:
14162306a36Sopenharmony_ci		da = 0x3;
14262306a36Sopenharmony_ci		break;
14362306a36Sopenharmony_ci	default:
14462306a36Sopenharmony_ci		return -EINVAL;
14562306a36Sopenharmony_ci	}
14662306a36Sopenharmony_ci	snd_soc_component_update_bits(component, MAX9850_DIGITAL_AUDIO, 0x3, da);
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci	return 0;
14962306a36Sopenharmony_ci}
15062306a36Sopenharmony_ci
15162306a36Sopenharmony_cistatic int max9850_set_dai_sysclk(struct snd_soc_dai *codec_dai,
15262306a36Sopenharmony_ci		int clk_id, unsigned int freq, int dir)
15362306a36Sopenharmony_ci{
15462306a36Sopenharmony_ci	struct snd_soc_component *component = codec_dai->component;
15562306a36Sopenharmony_ci	struct max9850_priv *max9850 = snd_soc_component_get_drvdata(component);
15662306a36Sopenharmony_ci
15762306a36Sopenharmony_ci	/* calculate mclk -> iclk divider */
15862306a36Sopenharmony_ci	if (freq <= 13000000)
15962306a36Sopenharmony_ci		snd_soc_component_write(component, MAX9850_CLOCK, 0x0);
16062306a36Sopenharmony_ci	else if (freq <= 26000000)
16162306a36Sopenharmony_ci		snd_soc_component_write(component, MAX9850_CLOCK, 0x4);
16262306a36Sopenharmony_ci	else if (freq <= 40000000)
16362306a36Sopenharmony_ci		snd_soc_component_write(component, MAX9850_CLOCK, 0x8);
16462306a36Sopenharmony_ci	else
16562306a36Sopenharmony_ci		return -EINVAL;
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_ci	max9850->sysclk = freq;
16862306a36Sopenharmony_ci	return 0;
16962306a36Sopenharmony_ci}
17062306a36Sopenharmony_ci
17162306a36Sopenharmony_cistatic int max9850_set_dai_fmt(struct snd_soc_dai *codec_dai, unsigned int fmt)
17262306a36Sopenharmony_ci{
17362306a36Sopenharmony_ci	struct snd_soc_component *component = codec_dai->component;
17462306a36Sopenharmony_ci	u8 da = 0;
17562306a36Sopenharmony_ci
17662306a36Sopenharmony_ci	/* set clock provider for audio interface */
17762306a36Sopenharmony_ci	switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
17862306a36Sopenharmony_ci	case SND_SOC_DAIFMT_CBP_CFP:
17962306a36Sopenharmony_ci		da |= MAX9850_MASTER;
18062306a36Sopenharmony_ci		break;
18162306a36Sopenharmony_ci	case SND_SOC_DAIFMT_CBC_CFC:
18262306a36Sopenharmony_ci		break;
18362306a36Sopenharmony_ci	default:
18462306a36Sopenharmony_ci		return -EINVAL;
18562306a36Sopenharmony_ci	}
18662306a36Sopenharmony_ci
18762306a36Sopenharmony_ci	/* interface format */
18862306a36Sopenharmony_ci	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
18962306a36Sopenharmony_ci	case SND_SOC_DAIFMT_I2S:
19062306a36Sopenharmony_ci		da |= MAX9850_DLY;
19162306a36Sopenharmony_ci		break;
19262306a36Sopenharmony_ci	case SND_SOC_DAIFMT_RIGHT_J:
19362306a36Sopenharmony_ci		da |= MAX9850_RTJ;
19462306a36Sopenharmony_ci		break;
19562306a36Sopenharmony_ci	case SND_SOC_DAIFMT_LEFT_J:
19662306a36Sopenharmony_ci		break;
19762306a36Sopenharmony_ci	default:
19862306a36Sopenharmony_ci		return -EINVAL;
19962306a36Sopenharmony_ci	}
20062306a36Sopenharmony_ci
20162306a36Sopenharmony_ci	/* clock inversion */
20262306a36Sopenharmony_ci	switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
20362306a36Sopenharmony_ci	case SND_SOC_DAIFMT_NB_NF:
20462306a36Sopenharmony_ci		break;
20562306a36Sopenharmony_ci	case SND_SOC_DAIFMT_IB_IF:
20662306a36Sopenharmony_ci		da |= MAX9850_BCINV | MAX9850_INV;
20762306a36Sopenharmony_ci		break;
20862306a36Sopenharmony_ci	case SND_SOC_DAIFMT_IB_NF:
20962306a36Sopenharmony_ci		da |= MAX9850_BCINV;
21062306a36Sopenharmony_ci		break;
21162306a36Sopenharmony_ci	case SND_SOC_DAIFMT_NB_IF:
21262306a36Sopenharmony_ci		da |= MAX9850_INV;
21362306a36Sopenharmony_ci		break;
21462306a36Sopenharmony_ci	default:
21562306a36Sopenharmony_ci		return -EINVAL;
21662306a36Sopenharmony_ci	}
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	/* set da */
21962306a36Sopenharmony_ci	snd_soc_component_write(component, MAX9850_DIGITAL_AUDIO, da);
22062306a36Sopenharmony_ci
22162306a36Sopenharmony_ci	return 0;
22262306a36Sopenharmony_ci}
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_cistatic int max9850_set_bias_level(struct snd_soc_component *component,
22562306a36Sopenharmony_ci				  enum snd_soc_bias_level level)
22662306a36Sopenharmony_ci{
22762306a36Sopenharmony_ci	struct max9850_priv *max9850 = snd_soc_component_get_drvdata(component);
22862306a36Sopenharmony_ci	int ret;
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	switch (level) {
23162306a36Sopenharmony_ci	case SND_SOC_BIAS_ON:
23262306a36Sopenharmony_ci		break;
23362306a36Sopenharmony_ci	case SND_SOC_BIAS_PREPARE:
23462306a36Sopenharmony_ci		break;
23562306a36Sopenharmony_ci	case SND_SOC_BIAS_STANDBY:
23662306a36Sopenharmony_ci		if (snd_soc_component_get_bias_level(component) == SND_SOC_BIAS_OFF) {
23762306a36Sopenharmony_ci			ret = regcache_sync(max9850->regmap);
23862306a36Sopenharmony_ci			if (ret) {
23962306a36Sopenharmony_ci				dev_err(component->dev,
24062306a36Sopenharmony_ci					"Failed to sync cache: %d\n", ret);
24162306a36Sopenharmony_ci				return ret;
24262306a36Sopenharmony_ci			}
24362306a36Sopenharmony_ci		}
24462306a36Sopenharmony_ci		break;
24562306a36Sopenharmony_ci	case SND_SOC_BIAS_OFF:
24662306a36Sopenharmony_ci		break;
24762306a36Sopenharmony_ci	}
24862306a36Sopenharmony_ci	return 0;
24962306a36Sopenharmony_ci}
25062306a36Sopenharmony_ci
25162306a36Sopenharmony_ci#define MAX9850_RATES SNDRV_PCM_RATE_8000_48000
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_ci#define MAX9850_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
25462306a36Sopenharmony_ci	SNDRV_PCM_FMTBIT_S24_LE)
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_cistatic const struct snd_soc_dai_ops max9850_dai_ops = {
25762306a36Sopenharmony_ci	.hw_params	= max9850_hw_params,
25862306a36Sopenharmony_ci	.set_sysclk	= max9850_set_dai_sysclk,
25962306a36Sopenharmony_ci	.set_fmt	= max9850_set_dai_fmt,
26062306a36Sopenharmony_ci};
26162306a36Sopenharmony_ci
26262306a36Sopenharmony_cistatic struct snd_soc_dai_driver max9850_dai = {
26362306a36Sopenharmony_ci	.name = "max9850-hifi",
26462306a36Sopenharmony_ci	.playback = {
26562306a36Sopenharmony_ci		.stream_name = "Playback",
26662306a36Sopenharmony_ci		.channels_min = 1,
26762306a36Sopenharmony_ci		.channels_max = 2,
26862306a36Sopenharmony_ci		.rates = MAX9850_RATES,
26962306a36Sopenharmony_ci		.formats = MAX9850_FORMATS
27062306a36Sopenharmony_ci	},
27162306a36Sopenharmony_ci	.ops = &max9850_dai_ops,
27262306a36Sopenharmony_ci};
27362306a36Sopenharmony_ci
27462306a36Sopenharmony_cistatic int max9850_probe(struct snd_soc_component *component)
27562306a36Sopenharmony_ci{
27662306a36Sopenharmony_ci	/* enable zero-detect */
27762306a36Sopenharmony_ci	snd_soc_component_update_bits(component, MAX9850_GENERAL_PURPOSE, 1, 1);
27862306a36Sopenharmony_ci	/* enable slew-rate control */
27962306a36Sopenharmony_ci	snd_soc_component_update_bits(component, MAX9850_VOLUME, 0x40, 0x40);
28062306a36Sopenharmony_ci	/* set slew-rate 125ms */
28162306a36Sopenharmony_ci	snd_soc_component_update_bits(component, MAX9850_CHARGE_PUMP, 0xff, 0xc0);
28262306a36Sopenharmony_ci
28362306a36Sopenharmony_ci	return 0;
28462306a36Sopenharmony_ci}
28562306a36Sopenharmony_ci
28662306a36Sopenharmony_cistatic const struct snd_soc_component_driver soc_component_dev_max9850 = {
28762306a36Sopenharmony_ci	.probe			= max9850_probe,
28862306a36Sopenharmony_ci	.set_bias_level		= max9850_set_bias_level,
28962306a36Sopenharmony_ci	.controls		= max9850_controls,
29062306a36Sopenharmony_ci	.num_controls		= ARRAY_SIZE(max9850_controls),
29162306a36Sopenharmony_ci	.dapm_widgets		= max9850_dapm_widgets,
29262306a36Sopenharmony_ci	.num_dapm_widgets	= ARRAY_SIZE(max9850_dapm_widgets),
29362306a36Sopenharmony_ci	.dapm_routes		= max9850_dapm_routes,
29462306a36Sopenharmony_ci	.num_dapm_routes	= ARRAY_SIZE(max9850_dapm_routes),
29562306a36Sopenharmony_ci	.suspend_bias_off	= 1,
29662306a36Sopenharmony_ci	.idle_bias_on		= 1,
29762306a36Sopenharmony_ci	.use_pmdown_time	= 1,
29862306a36Sopenharmony_ci	.endianness		= 1,
29962306a36Sopenharmony_ci};
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_cistatic int max9850_i2c_probe(struct i2c_client *i2c)
30262306a36Sopenharmony_ci{
30362306a36Sopenharmony_ci	struct max9850_priv *max9850;
30462306a36Sopenharmony_ci	int ret;
30562306a36Sopenharmony_ci
30662306a36Sopenharmony_ci	max9850 = devm_kzalloc(&i2c->dev, sizeof(struct max9850_priv),
30762306a36Sopenharmony_ci			       GFP_KERNEL);
30862306a36Sopenharmony_ci	if (max9850 == NULL)
30962306a36Sopenharmony_ci		return -ENOMEM;
31062306a36Sopenharmony_ci
31162306a36Sopenharmony_ci	max9850->regmap = devm_regmap_init_i2c(i2c, &max9850_regmap);
31262306a36Sopenharmony_ci	if (IS_ERR(max9850->regmap))
31362306a36Sopenharmony_ci		return PTR_ERR(max9850->regmap);
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_ci	i2c_set_clientdata(i2c, max9850);
31662306a36Sopenharmony_ci
31762306a36Sopenharmony_ci	ret = devm_snd_soc_register_component(&i2c->dev,
31862306a36Sopenharmony_ci			&soc_component_dev_max9850, &max9850_dai, 1);
31962306a36Sopenharmony_ci	return ret;
32062306a36Sopenharmony_ci}
32162306a36Sopenharmony_ci
32262306a36Sopenharmony_cistatic const struct i2c_device_id max9850_i2c_id[] = {
32362306a36Sopenharmony_ci	{ "max9850", 0 },
32462306a36Sopenharmony_ci	{ }
32562306a36Sopenharmony_ci};
32662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, max9850_i2c_id);
32762306a36Sopenharmony_ci
32862306a36Sopenharmony_cistatic struct i2c_driver max9850_i2c_driver = {
32962306a36Sopenharmony_ci	.driver = {
33062306a36Sopenharmony_ci		.name = "max9850",
33162306a36Sopenharmony_ci	},
33262306a36Sopenharmony_ci	.probe = max9850_i2c_probe,
33362306a36Sopenharmony_ci	.id_table = max9850_i2c_id,
33462306a36Sopenharmony_ci};
33562306a36Sopenharmony_ci
33662306a36Sopenharmony_cimodule_i2c_driver(max9850_i2c_driver);
33762306a36Sopenharmony_ci
33862306a36Sopenharmony_ciMODULE_AUTHOR("Christian Glindkamp <christian.glindkamp@taskit.de>");
33962306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC MAX9850 codec driver");
34062306a36Sopenharmony_ciMODULE_LICENSE("GPL");
341