Commit a07df82c authored by Olivia Mackintosh's avatar Olivia Mackintosh Committed by Takashi Iwai

ALSA: usb-audio: Add DJM750 to Pioneer mixer quirk

This allows for N different devices to use the pioneer mixer quirk for
setting capture/record type and recording level. The impementation has
not changed much with the exception of an additional mask on
private_value to allow storing of a device index:
	DEVICE MASK	0xff000000
	GROUP_MASK	0x00ff0000
	VALUE_MASK	0x0000ffff

This could be improved by changing the arrays of wValues for each
channel to contain named definitions (e.g. SND_DJM_CAP_LINE). It would
improve readability and perhaps would allow using the same array for
multiple channels. The channel number can be specified on the control
next to the wIndex.

Feedback is very much appreciated as I'm not the most proficient C
programmer but am learning as I go.
Signed-off-by: default avatarOlivia Mackintosh <livvy@base.nu>
Link: https://lore.kernel.org/r/20210205184256.10201-2-livvy@base.nuSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent fb3c293b
...@@ -2603,141 +2603,221 @@ static int snd_bbfpro_controls_create(struct usb_mixer_interface *mixer) ...@@ -2603,141 +2603,221 @@ static int snd_bbfpro_controls_create(struct usb_mixer_interface *mixer)
} }
/* /*
* Pioneer DJ DJM-250MK2 and maybe other DJM models * Pioneer DJ DJM Mixers
* *
* For playback, no duplicate mapping should be set. * These devices generally have options for soft-switching the playback and
* There are three mixer stereo channels (CH1, CH2, AUX) * capture sources in addition to the recording level. Although different
* and three stereo sources (Playback 1-2, Playback 3-4, Playback 5-6). * devices have different configurations, there seems to be canonical values
* Each channel should be mapped just once to one source. * for specific capture/playback types: See the definitions of these below.
* If mapped multiple times, only one source will play on given channel
* (sources are not mixed together).
* *
* For recording, duplicate mapping is OK. We will get the same signal multiple times. * The wValue is masked with the stereo channel number. e.g. Setting Ch2 to
* * capture phono would be 0x0203. Capture, playback and capture level have
* Channels 7-8 are in both directions fixed to FX SEND / FX RETURN. * different wIndexes.
*
* See also notes in the quirks-table.h file.
*/ */
struct snd_pioneer_djm_option { // Capture types
const u16 wIndex; #define SND_DJM_CAP_LINE 0x00
const u16 wValue; #define SND_DJM_CAP_CDLINE 0x01
#define SND_DJM_CAP_PHONO 0x03
#define SND_DJM_CAP_PFADER 0x06
#define SND_DJM_CAP_XFADERA 0x07
#define SND_DJM_CAP_XFADERB 0x08
#define SND_DJM_CAP_MIC 0x09
#define SND_DJM_CAP_AUX 0x0d
#define SND_DJM_CAP_RECOUT 0x0a
#define SND_DJM_CAP_NONE 0x0f
#define SND_DJM_CAP_CH1PFADER 0x11
#define SND_DJM_CAP_CH2PFADER 0x12
// Playback types
#define SND_DJM_PB_CH1 0x00
#define SND_DJM_PB_CH2 0x01
#define SND_DJM_PB_AUX 0x04
#define SND_DJM_WINDEX_CAP 0x8002
#define SND_DJM_WINDEX_CAPLVL 0x8003
#define SND_DJM_WINDEX_PB 0x8016
// kcontrol->private_value layout
#define SND_DJM_VALUE_MASK 0x0000ffff
#define SND_DJM_GROUP_MASK 0x00ff0000
#define SND_DJM_DEVICE_MASK 0xff000000
#define SND_DJM_GROUP_SHIFT 16
#define SND_DJM_DEVICE_SHIFT 24
// device table index
#define SND_DJM_250MK2_IDX 0x0
#define SND_DJM_750_IDX 0x1
#define SND_DJM_CTL(_name, suffix, _default_value, _windex) { \
.name = _name, \
.options = snd_djm_opts_##suffix, \
.noptions = ARRAY_SIZE(snd_djm_opts_##suffix), \
.default_value = _default_value, \
.wIndex = _windex }
#define SND_DJM_DEVICE(suffix) { \
.controls = snd_djm_ctls_##suffix, \
.ncontrols = ARRAY_SIZE(snd_djm_ctls_##suffix) }
struct snd_djm_device {
const char *name; const char *name;
const struct snd_djm_ctl *controls;
size_t ncontrols;
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_capture_level[] = { struct snd_djm_ctl {
{ .name = "-5 dB", .wValue = 0x0300, .wIndex = 0x8003 }, const char *name;
{ .name = "-10 dB", .wValue = 0x0200, .wIndex = 0x8003 }, const u16 *options;
{ .name = "-15 dB", .wValue = 0x0100, .wIndex = 0x8003 }, size_t noptions;
{ .name = "-19 dB", .wValue = 0x0000, .wIndex = 0x8003 } u16 default_value;
u16 wIndex;
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_capture_ch12[] = { static const char *snd_djm_get_label_caplevel(u16 wvalue)
{ .name = "CH1 Control Tone PHONO", .wValue = 0x0103, .wIndex = 0x8002 }, {
{ .name = "CH1 Control Tone LINE", .wValue = 0x0100, .wIndex = 0x8002 }, switch (wvalue) {
{ .name = "Post CH1 Fader", .wValue = 0x0106, .wIndex = 0x8002 }, case 0x0000: return "-19dB";
{ .name = "Cross Fader A", .wValue = 0x0107, .wIndex = 0x8002 }, case 0x0100: return "-15dB";
{ .name = "Cross Fader B", .wValue = 0x0108, .wIndex = 0x8002 }, case 0x0200: return "-10dB";
{ .name = "MIC", .wValue = 0x0109, .wIndex = 0x8002 }, case 0x0300: return "-5dB";
{ .name = "AUX", .wValue = 0x010d, .wIndex = 0x8002 }, default: return NULL;
{ .name = "REC OUT", .wValue = 0x010a, .wIndex = 0x8002 } }
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_capture_ch34[] = { static const char *snd_djm_get_label_cap(u16 wvalue)
{ .name = "CH2 Control Tone PHONO", .wValue = 0x0203, .wIndex = 0x8002 }, {
{ .name = "CH2 Control Tone LINE", .wValue = 0x0200, .wIndex = 0x8002 }, switch (wvalue & 0x00ff) {
{ .name = "Post CH2 Fader", .wValue = 0x0206, .wIndex = 0x8002 }, case SND_DJM_CAP_LINE: return "Control Tone LINE";
{ .name = "Cross Fader A", .wValue = 0x0207, .wIndex = 0x8002 }, case SND_DJM_CAP_CDLINE: return "Control Tone CD/LINE";
{ .name = "Cross Fader B", .wValue = 0x0208, .wIndex = 0x8002 }, case SND_DJM_CAP_PHONO: return "Control Tone PHONO";
{ .name = "MIC", .wValue = 0x0209, .wIndex = 0x8002 }, case SND_DJM_CAP_PFADER: return "Post Fader";
{ .name = "AUX", .wValue = 0x020d, .wIndex = 0x8002 }, case SND_DJM_CAP_XFADERA: return "Cross Fader A";
{ .name = "REC OUT", .wValue = 0x020a, .wIndex = 0x8002 } case SND_DJM_CAP_XFADERB: return "Cross Fader B";
case SND_DJM_CAP_MIC: return "Mic";
case SND_DJM_CAP_RECOUT: return "Rec Out";
case SND_DJM_CAP_AUX: return "Aux";
case SND_DJM_CAP_NONE: return "None";
case SND_DJM_CAP_CH1PFADER: return "Post Fader Ch1";
case SND_DJM_CAP_CH2PFADER: return "Post Fader Ch2";
default: return NULL;
}
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_capture_ch56[] = { static const char *snd_djm_get_label_pb(u16 wvalue)
{ .name = "REC OUT", .wValue = 0x030a, .wIndex = 0x8002 }, {
{ .name = "Post CH1 Fader", .wValue = 0x0311, .wIndex = 0x8002 }, switch (wvalue & 0x00ff) {
{ .name = "Post CH2 Fader", .wValue = 0x0312, .wIndex = 0x8002 }, case SND_DJM_PB_CH1: return "Ch1";
{ .name = "Cross Fader A", .wValue = 0x0307, .wIndex = 0x8002 }, case SND_DJM_PB_CH2: return "Ch2";
{ .name = "Cross Fader B", .wValue = 0x0308, .wIndex = 0x8002 }, case SND_DJM_PB_AUX: return "Aux";
{ .name = "MIC", .wValue = 0x0309, .wIndex = 0x8002 }, default: return NULL;
{ .name = "AUX", .wValue = 0x030d, .wIndex = 0x8002 } }
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_playback_12[] = { static const char *snd_djm_get_label(u16 wvalue, u16 windex)
{ .name = "CH1", .wValue = 0x0100, .wIndex = 0x8016 }, {
{ .name = "CH2", .wValue = 0x0101, .wIndex = 0x8016 }, switch (windex) {
{ .name = "AUX", .wValue = 0x0104, .wIndex = 0x8016 } case SND_DJM_WINDEX_CAPLVL: return snd_djm_get_label_caplevel(wvalue);
case SND_DJM_WINDEX_CAP: return snd_djm_get_label_cap(wvalue);
case SND_DJM_WINDEX_PB: return snd_djm_get_label_pb(wvalue);
default: return NULL;
}
}; };
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_playback_34[] = {
{ .name = "CH1", .wValue = 0x0200, .wIndex = 0x8016 },
{ .name = "CH2", .wValue = 0x0201, .wIndex = 0x8016 },
{ .name = "AUX", .wValue = 0x0204, .wIndex = 0x8016 }
};
static const struct snd_pioneer_djm_option snd_pioneer_djm_options_playback_56[] = { // DJM-250MK2
{ .name = "CH1", .wValue = 0x0300, .wIndex = 0x8016 }, static const u16 snd_djm_opts_cap_level[] = {
{ .name = "CH2", .wValue = 0x0301, .wIndex = 0x8016 }, 0x0000, 0x0100, 0x0200, 0x0300 };
{ .name = "AUX", .wValue = 0x0304, .wIndex = 0x8016 }
static const u16 snd_djm_opts_250mk2_cap1[] = {
0x0103, 0x0100, 0x0106, 0x0107, 0x0108, 0x0109, 0x010d, 0x010a };
static const u16 snd_djm_opts_250mk2_cap2[] = {
0x0203, 0x0200, 0x0206, 0x0207, 0x0208, 0x0209, 0x020d, 0x020a };
static const u16 snd_djm_opts_250mk2_cap3[] = {
0x030a, 0x0311, 0x0312, 0x0307, 0x0308, 0x0309, 0x030d };
static const u16 snd_djm_opts_250mk2_pb1[] = { 0x0100, 0x0101, 0x0104 };
static const u16 snd_djm_opts_250mk2_pb2[] = { 0x0200, 0x0201, 0x0204 };
static const u16 snd_djm_opts_250mk2_pb3[] = { 0x0300, 0x0301, 0x0304 };
static const struct snd_djm_ctl snd_djm_ctls_250mk2[] = {
SND_DJM_CTL("Capture Level", cap_level, 0, SND_DJM_WINDEX_CAPLVL),
SND_DJM_CTL("Ch1 Input", 250mk2_cap1, 2, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch2 Input", 250mk2_cap2, 2, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch3 Input", 250mk2_cap3, 0, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch1 Output", 250mk2_pb1, 0, SND_DJM_WINDEX_PB),
SND_DJM_CTL("Ch2 Output", 250mk2_pb2, 1, SND_DJM_WINDEX_PB),
SND_DJM_CTL("Ch3 Output", 250mk2_pb3, 2, SND_DJM_WINDEX_PB)
}; };
struct snd_pioneer_djm_option_group {
const char *name; // DJM-750
const struct snd_pioneer_djm_option *options; static const u16 snd_djm_opts_750_cap1[] = {
const size_t count; 0x0101, 0x0103, 0x0106, 0x0107, 0x0108, 0x0109, 0x010a, 0x010f };
const u16 default_value; static const u16 snd_djm_opts_750_cap2[] = {
0x0200, 0x0201, 0x0206, 0x0207, 0x0208, 0x0209, 0x020a, 0x020f };
static const u16 snd_djm_opts_750_cap3[] = {
0x0300, 0x0301, 0x0306, 0x0307, 0x0308, 0x0309, 0x030a, 0x030f };
static const u16 snd_djm_opts_750_cap4[] = {
0x0401, 0x0403, 0x0406, 0x0407, 0x0408, 0x0409, 0x040a, 0x040f };
static const struct snd_djm_ctl snd_djm_ctls_750[] = {
SND_DJM_CTL("Capture Level", cap_level, 0, SND_DJM_WINDEX_CAPLVL),
SND_DJM_CTL("Ch1 Input", 750_cap1, 2, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch2 Input", 750_cap2, 2, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch3 Input", 750_cap3, 0, SND_DJM_WINDEX_CAP),
SND_DJM_CTL("Ch4 Input", 750_cap4, 0, SND_DJM_WINDEX_CAP)
}; };
#define snd_pioneer_djm_option_group_item(_name, suffix, _default_value) { \
.name = _name, \ static const struct snd_djm_device snd_djm_devices[] = {
.options = snd_pioneer_djm_options_##suffix, \ SND_DJM_DEVICE(250mk2),
.count = ARRAY_SIZE(snd_pioneer_djm_options_##suffix), \ SND_DJM_DEVICE(750)
.default_value = _default_value }
static const struct snd_pioneer_djm_option_group snd_pioneer_djm_option_groups[] = {
snd_pioneer_djm_option_group_item("Master Capture Level Capture Switch", capture_level, 0),
snd_pioneer_djm_option_group_item("Capture 1-2 Capture Switch", capture_ch12, 2),
snd_pioneer_djm_option_group_item("Capture 3-4 Capture Switch", capture_ch34, 2),
snd_pioneer_djm_option_group_item("Capture 5-6 Capture Switch", capture_ch56, 0),
snd_pioneer_djm_option_group_item("Playback 1-2 Playback Switch", playback_12, 0),
snd_pioneer_djm_option_group_item("Playback 3-4 Playback Switch", playback_34, 1),
snd_pioneer_djm_option_group_item("Playback 5-6 Playback Switch", playback_56, 2)
}; };
// layout of the kcontrol->private_value:
#define SND_PIONEER_DJM_VALUE_MASK 0x0000ffff
#define SND_PIONEER_DJM_GROUP_MASK 0xffff0000
#define SND_PIONEER_DJM_GROUP_SHIFT 16
static int snd_pioneer_djm_controls_info(struct snd_kcontrol *kctl, struct snd_ctl_elem_info *info) static int snd_djm_controls_info(struct snd_kcontrol *kctl,
struct snd_ctl_elem_info *info)
{ {
u16 group_index = kctl->private_value >> SND_PIONEER_DJM_GROUP_SHIFT; unsigned long private_value = kctl->private_value;
size_t count; u8 device_idx = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
u8 ctl_idx = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
const struct snd_djm_device *device = &snd_djm_devices[device_idx];
const char *name; const char *name;
const struct snd_pioneer_djm_option_group *group; const struct snd_djm_ctl *ctl;
size_t noptions;
if (ctl_idx >= device->ncontrols)
return -EINVAL;
ctl = &device->controls[ctl_idx];
noptions = ctl->noptions;
if (info->value.enumerated.item >= noptions)
info->value.enumerated.item = noptions - 1;
if (group_index >= ARRAY_SIZE(snd_pioneer_djm_option_groups)) name = snd_djm_get_label(ctl->options[info->value.enumerated.item],
ctl->wIndex);
if (!name)
return -EINVAL; return -EINVAL;
group = &snd_pioneer_djm_option_groups[group_index];
count = group->count;
if (info->value.enumerated.item >= count)
info->value.enumerated.item = count - 1;
name = group->options[info->value.enumerated.item].name;
strscpy(info->value.enumerated.name, name, sizeof(info->value.enumerated.name)); strscpy(info->value.enumerated.name, name, sizeof(info->value.enumerated.name));
info->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED; info->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
info->count = 1; info->count = 1;
info->value.enumerated.items = count; info->value.enumerated.items = noptions;
return 0; return 0;
} }
static int snd_pioneer_djm_controls_update(struct usb_mixer_interface *mixer, u16 group, u16 value) static int snd_djm_controls_update(struct usb_mixer_interface *mixer,
u8 device_idx, u8 group, u16 value)
{ {
int err; int err;
const struct snd_djm_device *device = &snd_djm_devices[device_idx];
if (group >= ARRAY_SIZE(snd_pioneer_djm_option_groups) if ((group >= device->ncontrols) || value >= device->controls[group].noptions)
|| value >= snd_pioneer_djm_option_groups[group].count)
return -EINVAL; return -EINVAL;
err = snd_usb_lock_shutdown(mixer->chip); err = snd_usb_lock_shutdown(mixer->chip);
...@@ -2748,63 +2828,76 @@ static int snd_pioneer_djm_controls_update(struct usb_mixer_interface *mixer, u1 ...@@ -2748,63 +2828,76 @@ static int snd_pioneer_djm_controls_update(struct usb_mixer_interface *mixer, u1
mixer->chip->dev, usb_sndctrlpipe(mixer->chip->dev, 0), mixer->chip->dev, usb_sndctrlpipe(mixer->chip->dev, 0),
USB_REQ_SET_FEATURE, USB_REQ_SET_FEATURE,
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE, USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
snd_pioneer_djm_option_groups[group].options[value].wValue, device->controls[group].options[value],
snd_pioneer_djm_option_groups[group].options[value].wIndex, device->controls[group].wIndex,
NULL, 0); NULL, 0);
snd_usb_unlock_shutdown(mixer->chip); snd_usb_unlock_shutdown(mixer->chip);
return err; return err;
} }
static int snd_pioneer_djm_controls_get(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *elem) static int snd_djm_controls_get(struct snd_kcontrol *kctl,
struct snd_ctl_elem_value *elem)
{ {
elem->value.enumerated.item[0] = kctl->private_value & SND_PIONEER_DJM_VALUE_MASK; elem->value.enumerated.item[0] = kctl->private_value & SND_DJM_VALUE_MASK;
return 0; return 0;
} }
static int snd_pioneer_djm_controls_put(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *elem) static int snd_djm_controls_put(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *elem)
{ {
struct usb_mixer_elem_list *list = snd_kcontrol_chip(kctl); struct usb_mixer_elem_list *list = snd_kcontrol_chip(kctl);
struct usb_mixer_interface *mixer = list->mixer; struct usb_mixer_interface *mixer = list->mixer;
unsigned long private_value = kctl->private_value; unsigned long private_value = kctl->private_value;
u16 group = (private_value & SND_PIONEER_DJM_GROUP_MASK) >> SND_PIONEER_DJM_GROUP_SHIFT;
u8 device = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
u8 group = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
u16 value = elem->value.enumerated.item[0]; u16 value = elem->value.enumerated.item[0];
kctl->private_value = (group << SND_PIONEER_DJM_GROUP_SHIFT) | value; kctl->private_value = ((device << SND_DJM_DEVICE_SHIFT) |
(group << SND_DJM_GROUP_SHIFT) |
value);
return snd_pioneer_djm_controls_update(mixer, group, value); return snd_djm_controls_update(mixer, device, group, value);
} }
static int snd_pioneer_djm_controls_resume(struct usb_mixer_elem_list *list) static int snd_djm_controls_resume(struct usb_mixer_elem_list *list)
{ {
unsigned long private_value = list->kctl->private_value; unsigned long private_value = list->kctl->private_value;
u16 group = (private_value & SND_PIONEER_DJM_GROUP_MASK) >> SND_PIONEER_DJM_GROUP_SHIFT; u8 device = (private_value & SND_DJM_DEVICE_MASK) >> SND_DJM_DEVICE_SHIFT;
u16 value = (private_value & SND_PIONEER_DJM_VALUE_MASK); u8 group = (private_value & SND_DJM_GROUP_MASK) >> SND_DJM_GROUP_SHIFT;
u16 value = (private_value & SND_DJM_VALUE_MASK);
return snd_pioneer_djm_controls_update(list->mixer, group, value); return snd_djm_controls_update(list->mixer, device, group, value);
} }
static int snd_pioneer_djm_controls_create(struct usb_mixer_interface *mixer) static int snd_djm_controls_create(struct usb_mixer_interface *mixer,
const u8 device_idx)
{ {
int err, i; int err, i;
const struct snd_pioneer_djm_option_group *group; u16 value;
const struct snd_djm_device *device = &snd_djm_devices[device_idx];
struct snd_kcontrol_new knew = { struct snd_kcontrol_new knew = {
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
.access = SNDRV_CTL_ELEM_ACCESS_READWRITE, .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
.index = 0, .index = 0,
.info = snd_pioneer_djm_controls_info, .info = snd_djm_controls_info,
.get = snd_pioneer_djm_controls_get, .get = snd_djm_controls_get,
.put = snd_pioneer_djm_controls_put .put = snd_djm_controls_put
}; };
for (i = 0; i < ARRAY_SIZE(snd_pioneer_djm_option_groups); i++) { for (i = 0; i < device->ncontrols; i++) {
group = &snd_pioneer_djm_option_groups[i]; value = device->controls[i].default_value;
knew.name = group->name; knew.name = device->controls[i].name;
knew.private_value = (i << SND_PIONEER_DJM_GROUP_SHIFT) | group->default_value; knew.private_value = (
err = snd_pioneer_djm_controls_update(mixer, i, group->default_value); (device_idx << SND_DJM_DEVICE_SHIFT) |
(i << SND_DJM_GROUP_SHIFT) |
value);
err = snd_djm_controls_update(mixer, device_idx, i, value);
if (err) if (err)
return err; return err;
err = add_single_ctl_with_resume(mixer, 0, snd_pioneer_djm_controls_resume, err = add_single_ctl_with_resume(mixer, 0, snd_djm_controls_resume,
&knew, NULL); &knew, NULL);
if (err) if (err)
return err; return err;
...@@ -2917,7 +3010,10 @@ int snd_usb_mixer_apply_create_quirk(struct usb_mixer_interface *mixer) ...@@ -2917,7 +3010,10 @@ int snd_usb_mixer_apply_create_quirk(struct usb_mixer_interface *mixer)
err = snd_bbfpro_controls_create(mixer); err = snd_bbfpro_controls_create(mixer);
break; break;
case USB_ID(0x2b73, 0x0017): /* Pioneer DJ DJM-250MK2 */ case USB_ID(0x2b73, 0x0017): /* Pioneer DJ DJM-250MK2 */
err = snd_pioneer_djm_controls_create(mixer); err = snd_djm_controls_create(mixer, SND_DJM_250MK2_IDX);
break;
case USB_ID(0x08e4, 0x017f): /* Pioneer DJ DJM-750 */
err = snd_djm_controls_create(mixer, SND_DJM_750_IDX);
break; break;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment