12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114 |
- commit 9884107af289f8a441156b7a7eff73c39e74467e
- Author: Nicolas Chauvet <[email protected]>
- Date: Wed Mar 27 18:01:43 2024 +0100
- Add redhat patches
- diff --git a/arch/s390/include/asm/ipl.h b/arch/s390/include/asm/ipl.h
- index b0d00032479d..afb9544fb007 100644
- --- a/arch/s390/include/asm/ipl.h
- +++ b/arch/s390/include/asm/ipl.h
- @@ -139,6 +139,7 @@ int ipl_report_add_component(struct ipl_report *report, struct kexec_buf *kbuf,
- unsigned char flags, unsigned short cert);
- int ipl_report_add_certificate(struct ipl_report *report, void *key,
- unsigned long addr, unsigned long len);
- +bool ipl_get_secureboot(void);
-
- /*
- * DIAG 308 support
- diff --git a/arch/s390/kernel/ipl.c b/arch/s390/kernel/ipl.c
- index 8d0b95c17312..118ae555a179 100644
- --- a/arch/s390/kernel/ipl.c
- +++ b/arch/s390/kernel/ipl.c
- @@ -2520,3 +2520,8 @@ int ipl_report_free(struct ipl_report *report)
- }
-
- #endif
- +
- +bool ipl_get_secureboot(void)
- +{
- + return !!ipl_secure_flag;
- +}
- diff --git a/arch/s390/kernel/setup.c b/arch/s390/kernel/setup.c
- index de6ad0fb2328..5cc2758be027 100644
- --- a/arch/s390/kernel/setup.c
- +++ b/arch/s390/kernel/setup.c
- @@ -49,6 +49,7 @@
- #include <linux/memory.h>
- #include <linux/compat.h>
- #include <linux/start_kernel.h>
- +#include <linux/security.h>
- #include <linux/hugetlb.h>
- #include <linux/kmemleak.h>
-
- @@ -914,6 +915,9 @@ void __init setup_arch(char **cmdline_p)
-
- log_component_list();
-
- + if (ipl_get_secureboot())
- + security_lock_kernel_down("Secure IPL mode", LOCKDOWN_INTEGRITY_MAX);
- +
- /* Have one command line that is parsed and saved in /proc/cmdline */
- /* boot_command_line has been already set up in early.c */
- *cmdline_p = boot_command_line;
- diff --git a/arch/x86/kernel/setup.c b/arch/x86/kernel/setup.c
- index e63a8d05ce29..92700cd1048e 100644
- --- a/arch/x86/kernel/setup.c
- +++ b/arch/x86/kernel/setup.c
- @@ -20,6 +20,7 @@
- #include <linux/root_dev.h>
- #include <linux/hugetlb.h>
- #include <linux/tboot.h>
- +#include <linux/security.h>
- #include <linux/usb/xhci-dbgp.h>
- #include <linux/static_call.h>
- #include <linux/swiotlb.h>
- @@ -1027,6 +1028,13 @@ void __init setup_arch(char **cmdline_p)
- if (efi_enabled(EFI_BOOT))
- efi_init();
-
- + efi_set_secure_boot(boot_params.secure_boot);
- +
- +#ifdef CONFIG_LOCK_DOWN_IN_EFI_SECURE_BOOT
- + if (efi_enabled(EFI_SECURE_BOOT))
- + security_lock_kernel_down("EFI Secure Boot mode", LOCKDOWN_INTEGRITY_MAX);
- +#endif
- +
- reserve_ibft_region();
- x86_init.resources.dmi_setup();
-
- @@ -1188,19 +1196,7 @@ void __init setup_arch(char **cmdline_p)
- /* Allocate bigger log buffer */
- setup_log_buf(1);
-
- - if (efi_enabled(EFI_BOOT)) {
- - switch (boot_params.secure_boot) {
- - case efi_secureboot_mode_disabled:
- - pr_info("Secure boot disabled\n");
- - break;
- - case efi_secureboot_mode_enabled:
- - pr_info("Secure boot enabled\n");
- - break;
- - default:
- - pr_info("Secure boot could not be determined\n");
- - break;
- - }
- - }
- + efi_set_secure_boot(boot_params.secure_boot);
-
- reserve_initrd();
-
- diff --git a/drivers/acpi/apei/hest.c b/drivers/acpi/apei/hest.c
- index 6aef1ee5e1bd..8f146b1b4972 100644
- --- a/drivers/acpi/apei/hest.c
- +++ b/drivers/acpi/apei/hest.c
- @@ -96,6 +96,14 @@ static int apei_hest_parse(apei_hest_func_t func, void *data)
- if (hest_disable || !hest_tab)
- return -EINVAL;
-
- +#ifdef CONFIG_ARM64
- + /* Ignore broken firmware */
- + if (!strncmp(hest_tab->header.oem_id, "HPE ", 6) &&
- + !strncmp(hest_tab->header.oem_table_id, "ProLiant", 8) &&
- + MIDR_IMPLEMENTOR(read_cpuid_id()) == ARM_CPU_IMP_APM)
- + return -EINVAL;
- +#endif
- +
- hest_hdr = (struct acpi_hest_header *)(hest_tab + 1);
- for (i = 0; i < hest_tab->error_source_count; i++) {
- len = hest_esrc_len(hest_hdr);
- diff --git a/drivers/acpi/irq.c b/drivers/acpi/irq.c
- index 1687483ff319..390b67f19181 100644
- --- a/drivers/acpi/irq.c
- +++ b/drivers/acpi/irq.c
- @@ -143,6 +143,7 @@ struct acpi_irq_parse_one_ctx {
- unsigned int index;
- unsigned long *res_flags;
- struct irq_fwspec *fwspec;
- + bool skip_producer_check;
- };
-
- /**
- @@ -216,7 +217,8 @@ static acpi_status acpi_irq_parse_one_cb(struct acpi_resource *ares,
- return AE_CTRL_TERMINATE;
- case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
- eirq = &ares->data.extended_irq;
- - if (eirq->producer_consumer == ACPI_PRODUCER)
- + if (!ctx->skip_producer_check &&
- + eirq->producer_consumer == ACPI_PRODUCER)
- return AE_OK;
- if (ctx->index >= eirq->interrupt_count) {
- ctx->index -= eirq->interrupt_count;
- @@ -252,8 +254,19 @@ static acpi_status acpi_irq_parse_one_cb(struct acpi_resource *ares,
- static int acpi_irq_parse_one(acpi_handle handle, unsigned int index,
- struct irq_fwspec *fwspec, unsigned long *flags)
- {
- - struct acpi_irq_parse_one_ctx ctx = { -EINVAL, index, flags, fwspec };
- + struct acpi_irq_parse_one_ctx ctx = { -EINVAL, index, flags, fwspec, false };
-
- + /*
- + * Firmware on arm64-based HPE m400 platform incorrectly marks
- + * its UART interrupt as ACPI_PRODUCER rather than ACPI_CONSUMER.
- + * Don't do the producer/consumer check for that device.
- + */
- + if (IS_ENABLED(CONFIG_ARM64)) {
- + struct acpi_device *adev = acpi_get_acpi_dev(handle);
- +
- + if (adev && !strcmp(acpi_device_hid(adev), "APMC0D08"))
- + ctx.skip_producer_check = true;
- + }
- acpi_walk_resources(handle, METHOD_NAME__CRS, acpi_irq_parse_one_cb, &ctx);
- return ctx.rc;
- }
- diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c
- index 597d75baa1cc..ee38441143bf 100644
- --- a/drivers/acpi/scan.c
- +++ b/drivers/acpi/scan.c
- @@ -1753,6 +1753,15 @@ static bool acpi_device_enumeration_by_parent(struct acpi_device *device)
- if (!acpi_match_device_ids(device, ignore_serial_bus_ids))
- return false;
-
- + /*
- + * Firmware on some arm64 X-Gene platforms will make the UART
- + * device appear as both a UART and a slave of that UART. Just
- + * bail out here for X-Gene UARTs.
- + */
- + if (IS_ENABLED(CONFIG_ARM64) &&
- + !strcmp(acpi_device_hid(device), "APMC0D08"))
- + return false;
- +
- INIT_LIST_HEAD(&resource_list);
- acpi_dev_get_resources(device, &resource_list,
- acpi_check_serial_bus_slave,
- diff --git a/drivers/ata/libahci.c b/drivers/ata/libahci.c
- index f1263364fa97..24ac410f4366 100644
- --- a/drivers/ata/libahci.c
- +++ b/drivers/ata/libahci.c
- @@ -729,6 +729,24 @@ int ahci_stop_engine(struct ata_port *ap)
- tmp &= ~PORT_CMD_START;
- writel(tmp, port_mmio + PORT_CMD);
-
- +#ifdef CONFIG_ARM64
- + /* Rev Ax of Cavium CN99XX needs a hack for port stop */
- + if (dev_is_pci(ap->host->dev) &&
- + to_pci_dev(ap->host->dev)->vendor == 0x14e4 &&
- + to_pci_dev(ap->host->dev)->device == 0x9027 &&
- + midr_is_cpu_model_range(read_cpuid_id(),
- + MIDR_CPU_MODEL(ARM_CPU_IMP_BRCM, BRCM_CPU_PART_VULCAN),
- + MIDR_CPU_VAR_REV(0, 0),
- + MIDR_CPU_VAR_REV(0, MIDR_REVISION_MASK))) {
- + tmp = readl(hpriv->mmio + 0x8000);
- + udelay(100);
- + writel(tmp | (1 << 26), hpriv->mmio + 0x8000);
- + udelay(100);
- + writel(tmp & ~(1 << 26), hpriv->mmio + 0x8000);
- + dev_warn(ap->host->dev, "CN99XX SATA reset workaround applied\n");
- + }
- +#endif
- +
- /* wait for engine to stop. This could be as long as 500 msec */
- tmp = ata_wait_register(ap, port_mmio + PORT_CMD,
- PORT_CMD_LIST_ON, PORT_CMD_LIST_ON, 1, 500);
- diff --git a/drivers/char/ipmi/ipmi_dmi.c b/drivers/char/ipmi/ipmi_dmi.c
- index bbf7029e224b..cf7faa970dd6 100644
- --- a/drivers/char/ipmi/ipmi_dmi.c
- +++ b/drivers/char/ipmi/ipmi_dmi.c
- @@ -215,6 +215,21 @@ static int __init scan_for_dmi_ipmi(void)
- {
- const struct dmi_device *dev = NULL;
-
- +#ifdef CONFIG_ARM64
- + /* RHEL-only
- + * If this is ARM-based HPE m400, return now, because that platform
- + * reports the host-side ipmi address as intel port-io space, which
- + * does not exist in the ARM architecture.
- + */
- + const char *dmistr = dmi_get_system_info(DMI_PRODUCT_NAME);
- +
- + if (dmistr && (strcmp("ProLiant m400 Server", dmistr) == 0)) {
- + pr_debug("%s does not support host ipmi\n", dmistr);
- + return 0;
- + }
- + /* END RHEL-only */
- +#endif
- +
- while ((dev = dmi_find_device(DMI_DEV_TYPE_IPMI, NULL, dev)))
- dmi_decode_ipmi((const struct dmi_header *) dev->device_data);
-
- diff --git a/drivers/char/ipmi/ipmi_msghandler.c b/drivers/char/ipmi/ipmi_msghandler.c
- index 186f1fee7534..93e3a76596ff 100644
- --- a/drivers/char/ipmi/ipmi_msghandler.c
- +++ b/drivers/char/ipmi/ipmi_msghandler.c
- @@ -35,6 +35,7 @@
- #include <linux/uuid.h>
- #include <linux/nospec.h>
- #include <linux/vmalloc.h>
- +#include <linux/dmi.h>
- #include <linux/delay.h>
-
- #define IPMI_DRIVER_VERSION "39.2"
- @@ -5516,8 +5517,21 @@ static int __init ipmi_init_msghandler_mod(void)
- {
- int rv;
-
- - pr_info("version " IPMI_DRIVER_VERSION "\n");
- +#ifdef CONFIG_ARM64
- + /* RHEL-only
- + * If this is ARM-based HPE m400, return now, because that platform
- + * reports the host-side ipmi address as intel port-io space, which
- + * does not exist in the ARM architecture.
- + */
- + const char *dmistr = dmi_get_system_info(DMI_PRODUCT_NAME);
-
- + if (dmistr && (strcmp("ProLiant m400 Server", dmistr) == 0)) {
- + pr_debug("%s does not support host ipmi\n", dmistr);
- + return -ENOSYS;
- + }
- + /* END RHEL-only */
- +#endif
- + pr_info("version " IPMI_DRIVER_VERSION "\n");
- mutex_lock(&ipmi_interfaces_mutex);
- rv = ipmi_register_driver();
- mutex_unlock(&ipmi_interfaces_mutex);
- diff --git a/drivers/firmware/efi/Makefile b/drivers/firmware/efi/Makefile
- index e489fefd23da..f2dfae764fb5 100644
- --- a/drivers/firmware/efi/Makefile
- +++ b/drivers/firmware/efi/Makefile
- @@ -25,6 +25,7 @@ subdir-$(CONFIG_EFI_STUB) += libstub
- obj-$(CONFIG_EFI_BOOTLOADER_CONTROL) += efibc.o
- obj-$(CONFIG_EFI_TEST) += test/
- obj-$(CONFIG_EFI_DEV_PATH_PARSER) += dev-path-parser.o
- +obj-$(CONFIG_EFI) += secureboot.o
- obj-$(CONFIG_APPLE_PROPERTIES) += apple-properties.o
- obj-$(CONFIG_EFI_RCI2_TABLE) += rci2-table.o
- obj-$(CONFIG_EFI_EMBEDDED_FIRMWARE) += embedded-firmware.o
- diff --git a/drivers/firmware/efi/efi.c b/drivers/firmware/efi/efi.c
- index 2c1095dcc2f2..7ea6012e3ef4 100644
- --- a/drivers/firmware/efi/efi.c
- +++ b/drivers/firmware/efi/efi.c
- @@ -32,6 +32,7 @@
- #include <linux/ucs2_string.h>
- #include <linux/memblock.h>
- #include <linux/security.h>
- +#include <linux/bsearch.h>
-
- #include <asm/early_ioremap.h>
-
- @@ -985,40 +986,101 @@ int efi_mem_type(unsigned long phys_addr)
- }
- #endif
-
- +struct efi_error_code {
- + efi_status_t status;
- + int errno;
- + const char *description;
- +};
- +
- +static const struct efi_error_code efi_error_codes[] = {
- + { EFI_SUCCESS, 0, "Success"},
- +#if 0
- + { EFI_LOAD_ERROR, -EPICK_AN_ERRNO, "Load Error"},
- +#endif
- + { EFI_INVALID_PARAMETER, -EINVAL, "Invalid Parameter"},
- + { EFI_UNSUPPORTED, -ENOSYS, "Unsupported"},
- + { EFI_BAD_BUFFER_SIZE, -ENOSPC, "Bad Buffer Size"},
- + { EFI_BUFFER_TOO_SMALL, -ENOSPC, "Buffer Too Small"},
- + { EFI_NOT_READY, -EAGAIN, "Not Ready"},
- + { EFI_DEVICE_ERROR, -EIO, "Device Error"},
- + { EFI_WRITE_PROTECTED, -EROFS, "Write Protected"},
- + { EFI_OUT_OF_RESOURCES, -ENOMEM, "Out of Resources"},
- +#if 0
- + { EFI_VOLUME_CORRUPTED, -EPICK_AN_ERRNO, "Volume Corrupt"},
- + { EFI_VOLUME_FULL, -EPICK_AN_ERRNO, "Volume Full"},
- + { EFI_NO_MEDIA, -EPICK_AN_ERRNO, "No Media"},
- + { EFI_MEDIA_CHANGED, -EPICK_AN_ERRNO, "Media changed"},
- +#endif
- + { EFI_NOT_FOUND, -ENOENT, "Not Found"},
- +#if 0
- + { EFI_ACCESS_DENIED, -EPICK_AN_ERRNO, "Access Denied"},
- + { EFI_NO_RESPONSE, -EPICK_AN_ERRNO, "No Response"},
- + { EFI_NO_MAPPING, -EPICK_AN_ERRNO, "No mapping"},
- + { EFI_TIMEOUT, -EPICK_AN_ERRNO, "Time out"},
- + { EFI_NOT_STARTED, -EPICK_AN_ERRNO, "Not started"},
- + { EFI_ALREADY_STARTED, -EPICK_AN_ERRNO, "Already started"},
- +#endif
- + { EFI_ABORTED, -EINTR, "Aborted"},
- +#if 0
- + { EFI_ICMP_ERROR, -EPICK_AN_ERRNO, "ICMP Error"},
- + { EFI_TFTP_ERROR, -EPICK_AN_ERRNO, "TFTP Error"},
- + { EFI_PROTOCOL_ERROR, -EPICK_AN_ERRNO, "Protocol Error"},
- + { EFI_INCOMPATIBLE_VERSION, -EPICK_AN_ERRNO, "Incompatible Version"},
- +#endif
- + { EFI_SECURITY_VIOLATION, -EACCES, "Security Policy Violation"},
- +#if 0
- + { EFI_CRC_ERROR, -EPICK_AN_ERRNO, "CRC Error"},
- + { EFI_END_OF_MEDIA, -EPICK_AN_ERRNO, "End of Media"},
- + { EFI_END_OF_FILE, -EPICK_AN_ERRNO, "End of File"},
- + { EFI_INVALID_LANGUAGE, -EPICK_AN_ERRNO, "Invalid Languages"},
- + { EFI_COMPROMISED_DATA, -EPICK_AN_ERRNO, "Compromised Data"},
- +
- + // warnings
- + { EFI_WARN_UNKOWN_GLYPH, -EPICK_AN_ERRNO, "Warning Unknown Glyph"},
- + { EFI_WARN_DELETE_FAILURE, -EPICK_AN_ERRNO, "Warning Delete Failure"},
- + { EFI_WARN_WRITE_FAILURE, -EPICK_AN_ERRNO, "Warning Write Failure"},
- + { EFI_WARN_BUFFER_TOO_SMALL, -EPICK_AN_ERRNO, "Warning Buffer Too Small"},
- +#endif
- +};
- +
- +static int
- +efi_status_cmp_bsearch(const void *key, const void *item)
- +{
- + u64 status = (u64)(uintptr_t)key;
- + struct efi_error_code *code = (struct efi_error_code *)item;
- +
- + if (status < code->status)
- + return -1;
- + if (status > code->status)
- + return 1;
- + return 0;
- +}
- +
- int efi_status_to_err(efi_status_t status)
- {
- - int err;
- -
- - switch (status) {
- - case EFI_SUCCESS:
- - err = 0;
- - break;
- - case EFI_INVALID_PARAMETER:
- - err = -EINVAL;
- - break;
- - case EFI_OUT_OF_RESOURCES:
- - err = -ENOSPC;
- - break;
- - case EFI_DEVICE_ERROR:
- - err = -EIO;
- - break;
- - case EFI_WRITE_PROTECTED:
- - err = -EROFS;
- - break;
- - case EFI_SECURITY_VIOLATION:
- - err = -EACCES;
- - break;
- - case EFI_NOT_FOUND:
- - err = -ENOENT;
- - break;
- - case EFI_ABORTED:
- - err = -EINTR;
- - break;
- - default:
- - err = -EINVAL;
- - }
- + struct efi_error_code *found;
- + size_t num = sizeof(efi_error_codes) / sizeof(struct efi_error_code);
-
- - return err;
- + found = bsearch((void *)(uintptr_t)status, efi_error_codes,
- + sizeof(struct efi_error_code), num,
- + efi_status_cmp_bsearch);
- + if (!found)
- + return -EINVAL;
- + return found->errno;
- +}
- +
- +const char *
- +efi_status_to_str(efi_status_t status)
- +{
- + struct efi_error_code *found;
- + size_t num = sizeof(efi_error_codes) / sizeof(struct efi_error_code);
- +
- + found = bsearch((void *)(uintptr_t)status, efi_error_codes,
- + sizeof(struct efi_error_code), num,
- + efi_status_cmp_bsearch);
- + if (!found)
- + return "Unknown error code";
- + return found->description;
- }
- EXPORT_SYMBOL_GPL(efi_status_to_err);
-
- diff --git a/drivers/firmware/efi/secureboot.c b/drivers/firmware/efi/secureboot.c
- new file mode 100644
- index 000000000000..de0a3714a5d4
- --- /dev/null
- +++ b/drivers/firmware/efi/secureboot.c
- @@ -0,0 +1,38 @@
- +/* Core kernel secure boot support.
- + *
- + * Copyright (C) 2017 Red Hat, Inc. All Rights Reserved.
- + * Written by David Howells ([email protected])
- + *
- + * This program is free software; you can redistribute it and/or
- + * modify it under the terms of the GNU General Public Licence
- + * as published by the Free Software Foundation; either version
- + * 2 of the Licence, or (at your option) any later version.
- + */
- +
- +#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- +
- +#include <linux/efi.h>
- +#include <linux/kernel.h>
- +#include <linux/printk.h>
- +
- +/*
- + * Decide what to do when UEFI secure boot mode is enabled.
- + */
- +void __init efi_set_secure_boot(enum efi_secureboot_mode mode)
- +{
- + if (efi_enabled(EFI_BOOT)) {
- + switch (mode) {
- + case efi_secureboot_mode_disabled:
- + pr_info("Secure boot disabled\n");
- + break;
- + case efi_secureboot_mode_enabled:
- + set_bit(EFI_SECURE_BOOT, &efi.flags);
- + pr_info("Secure boot enabled\n");
- + break;
- + default:
- + pr_warn("Secure boot could not be determined (mode %u)\n",
- + mode);
- + break;
- + }
- + }
- +}
- diff --git a/drivers/firmware/sysfb.c b/drivers/firmware/sysfb.c
- index 3c197db42c9d..16e4a2e90fae 100644
- --- a/drivers/firmware/sysfb.c
- +++ b/drivers/firmware/sysfb.c
- @@ -34,6 +34,22 @@
- #include <linux/screen_info.h>
- #include <linux/sysfb.h>
-
- +static int skip_simpledrm;
- +
- +static int __init simpledrm_disable(char *opt)
- +{
- + if (!opt)
- + return -EINVAL;
- +
- + get_option(&opt, &skip_simpledrm);
- +
- + if (skip_simpledrm)
- + pr_info("The simpledrm driver will not be probed\n");
- +
- + return 0;
- +}
- +early_param("nvidia-drm.modeset", simpledrm_disable);
- +
- static struct platform_device *pd;
- static DEFINE_MUTEX(disable_lock);
- static bool disabled;
- @@ -85,7 +101,7 @@ static __init int sysfb_init(void)
-
- /* try to create a simple-framebuffer device */
- compatible = sysfb_parse_mode(si, &mode);
- - if (compatible) {
- + if (compatible && !skip_simpledrm) {
- pd = sysfb_create_simplefb(si, &mode);
- if (!IS_ERR(pd))
- goto unlock_mutex;
- diff --git a/drivers/hid/hid-rmi.c b/drivers/hid/hid-rmi.c
- index 84e7ba5314d3..efc96776f761 100644
- --- a/drivers/hid/hid-rmi.c
- +++ b/drivers/hid/hid-rmi.c
- @@ -321,21 +321,12 @@ static int rmi_input_event(struct hid_device *hdev, u8 *data, int size)
- {
- struct rmi_data *hdata = hid_get_drvdata(hdev);
- struct rmi_device *rmi_dev = hdata->xport.rmi_dev;
- - unsigned long flags;
-
- if (!(test_bit(RMI_STARTED, &hdata->flags)))
- return 0;
-
- - pm_wakeup_event(hdev->dev.parent, 0);
- -
- - local_irq_save(flags);
- -
- rmi_set_attn_data(rmi_dev, data[1], &data[2], size - 2);
-
- - generic_handle_irq(hdata->rmi_irq);
- -
- - local_irq_restore(flags);
- -
- return 1;
- }
-
- @@ -591,56 +582,6 @@ static const struct rmi_transport_ops hid_rmi_ops = {
- .reset = rmi_hid_reset,
- };
-
- -static void rmi_irq_teardown(void *data)
- -{
- - struct rmi_data *hdata = data;
- - struct irq_domain *domain = hdata->domain;
- -
- - if (!domain)
- - return;
- -
- - irq_dispose_mapping(irq_find_mapping(domain, 0));
- -
- - irq_domain_remove(domain);
- - hdata->domain = NULL;
- - hdata->rmi_irq = 0;
- -}
- -
- -static int rmi_irq_map(struct irq_domain *h, unsigned int virq,
- - irq_hw_number_t hw_irq_num)
- -{
- - irq_set_chip_and_handler(virq, &dummy_irq_chip, handle_simple_irq);
- -
- - return 0;
- -}
- -
- -static const struct irq_domain_ops rmi_irq_ops = {
- - .map = rmi_irq_map,
- -};
- -
- -static int rmi_setup_irq_domain(struct hid_device *hdev)
- -{
- - struct rmi_data *hdata = hid_get_drvdata(hdev);
- - int ret;
- -
- - hdata->domain = irq_domain_create_linear(hdev->dev.fwnode, 1,
- - &rmi_irq_ops, hdata);
- - if (!hdata->domain)
- - return -ENOMEM;
- -
- - ret = devm_add_action_or_reset(&hdev->dev, &rmi_irq_teardown, hdata);
- - if (ret)
- - return ret;
- -
- - hdata->rmi_irq = irq_create_mapping(hdata->domain, 0);
- - if (hdata->rmi_irq <= 0) {
- - hid_err(hdev, "Can't allocate an IRQ\n");
- - return hdata->rmi_irq < 0 ? hdata->rmi_irq : -ENXIO;
- - }
- -
- - return 0;
- -}
- -
- static int rmi_probe(struct hid_device *hdev, const struct hid_device_id *id)
- {
- struct rmi_data *data = NULL;
- @@ -713,18 +654,11 @@ static int rmi_probe(struct hid_device *hdev, const struct hid_device_id *id)
-
- mutex_init(&data->page_mutex);
-
- - ret = rmi_setup_irq_domain(hdev);
- - if (ret) {
- - hid_err(hdev, "failed to allocate IRQ domain\n");
- - return ret;
- - }
- -
- if (data->device_flags & RMI_DEVICE_HAS_PHYS_BUTTONS)
- rmi_hid_pdata.gpio_data.disable = true;
-
- data->xport.dev = hdev->dev.parent;
- data->xport.pdata = rmi_hid_pdata;
- - data->xport.pdata.irq = data->rmi_irq;
- data->xport.proto_name = "hid";
- data->xport.ops = &hid_rmi_ops;
-
- diff --git a/drivers/hwtracing/coresight/coresight-etm4x-core.c b/drivers/hwtracing/coresight/coresight-etm4x-core.c
- index 18c4544f6045..4c37cdc40426 100644
- --- a/drivers/hwtracing/coresight/coresight-etm4x-core.c
- +++ b/drivers/hwtracing/coresight/coresight-etm4x-core.c
- @@ -10,6 +10,7 @@
- #include <linux/init.h>
- #include <linux/types.h>
- #include <linux/device.h>
- +#include <linux/dmi.h>
- #include <linux/io.h>
- #include <linux/err.h>
- #include <linux/fs.h>
- @@ -2303,6 +2304,16 @@ static const struct amba_id etm4_ids[] = {
- {},
- };
-
- +static const struct dmi_system_id broken_coresight[] = {
- + {
- + .matches = {
- + DMI_MATCH(DMI_SYS_VENDOR, "HPE"),
- + DMI_MATCH(DMI_PRODUCT_NAME, "Apollo 70"),
- + },
- + },
- + { } /* terminating entry */
- +};
- +
- MODULE_DEVICE_TABLE(amba, etm4_ids);
-
- static struct amba_driver etm4x_amba_driver = {
- @@ -2372,6 +2383,11 @@ static int __init etm4x_init(void)
- {
- int ret;
-
- + if (dmi_check_system(broken_coresight)) {
- + pr_info("ETM4 disabled due to firmware bug\n");
- + return 0;
- + }
- +
- ret = etm4_pm_setup();
-
- /* etm4_pm_setup() does its own cleanup - exit on error */
- @@ -2398,6 +2414,9 @@ static int __init etm4x_init(void)
-
- static void __exit etm4x_exit(void)
- {
- + if (dmi_check_system(broken_coresight))
- + return;
- +
- amba_driver_unregister(&etm4x_amba_driver);
- platform_driver_unregister(&etm4_platform_driver);
- etm4_pm_clear();
- diff --git a/drivers/input/rmi4/rmi_driver.c b/drivers/input/rmi4/rmi_driver.c
- index 258d5fe3d395..f7298e3dc8f3 100644
- --- a/drivers/input/rmi4/rmi_driver.c
- +++ b/drivers/input/rmi4/rmi_driver.c
- @@ -182,34 +182,47 @@ void rmi_set_attn_data(struct rmi_device *rmi_dev, unsigned long irq_status,
- attn_data.data = fifo_data;
-
- kfifo_put(&drvdata->attn_fifo, attn_data);
- +
- + schedule_work(&drvdata->attn_work);
- }
- EXPORT_SYMBOL_GPL(rmi_set_attn_data);
-
- -static irqreturn_t rmi_irq_fn(int irq, void *dev_id)
- +static void attn_callback(struct work_struct *work)
- {
- - struct rmi_device *rmi_dev = dev_id;
- - struct rmi_driver_data *drvdata = dev_get_drvdata(&rmi_dev->dev);
- + struct rmi_driver_data *drvdata = container_of(work,
- + struct rmi_driver_data,
- + attn_work);
- struct rmi4_attn_data attn_data = {0};
- int ret, count;
-
- count = kfifo_get(&drvdata->attn_fifo, &attn_data);
- - if (count) {
- - *(drvdata->irq_status) = attn_data.irq_status;
- - drvdata->attn_data = attn_data;
- - }
- + if (!count)
- + return;
-
- - ret = rmi_process_interrupt_requests(rmi_dev);
- + *(drvdata->irq_status) = attn_data.irq_status;
- + drvdata->attn_data = attn_data;
- +
- + ret = rmi_process_interrupt_requests(drvdata->rmi_dev);
- if (ret)
- - rmi_dbg(RMI_DEBUG_CORE, &rmi_dev->dev,
- + rmi_dbg(RMI_DEBUG_CORE, &drvdata->rmi_dev->dev,
- "Failed to process interrupt request: %d\n", ret);
-
- - if (count) {
- - kfree(attn_data.data);
- - drvdata->attn_data.data = NULL;
- - }
- + kfree(attn_data.data);
- + drvdata->attn_data.data = NULL;
-
- if (!kfifo_is_empty(&drvdata->attn_fifo))
- - return rmi_irq_fn(irq, dev_id);
- + schedule_work(&drvdata->attn_work);
- +}
- +
- +static irqreturn_t rmi_irq_fn(int irq, void *dev_id)
- +{
- + struct rmi_device *rmi_dev = dev_id;
- + int ret;
- +
- + ret = rmi_process_interrupt_requests(rmi_dev);
- + if (ret)
- + rmi_dbg(RMI_DEBUG_CORE, &rmi_dev->dev,
- + "Failed to process interrupt request: %d\n", ret);
-
- return IRQ_HANDLED;
- }
- @@ -217,7 +230,6 @@ static irqreturn_t rmi_irq_fn(int irq, void *dev_id)
- static int rmi_irq_init(struct rmi_device *rmi_dev)
- {
- struct rmi_device_platform_data *pdata = rmi_get_platform_data(rmi_dev);
- - struct rmi_driver_data *data = dev_get_drvdata(&rmi_dev->dev);
- int irq_flags = irq_get_trigger_type(pdata->irq);
- int ret;
-
- @@ -235,8 +247,6 @@ static int rmi_irq_init(struct rmi_device *rmi_dev)
- return ret;
- }
-
- - data->enabled = true;
- -
- return 0;
- }
-
- @@ -886,23 +896,27 @@ void rmi_enable_irq(struct rmi_device *rmi_dev, bool clear_wake)
- if (data->enabled)
- goto out;
-
- - enable_irq(irq);
- - data->enabled = true;
- - if (clear_wake && device_may_wakeup(rmi_dev->xport->dev)) {
- - retval = disable_irq_wake(irq);
- - if (retval)
- - dev_warn(&rmi_dev->dev,
- - "Failed to disable irq for wake: %d\n",
- - retval);
- - }
- + if (irq) {
- + enable_irq(irq);
- + data->enabled = true;
- + if (clear_wake && device_may_wakeup(rmi_dev->xport->dev)) {
- + retval = disable_irq_wake(irq);
- + if (retval)
- + dev_warn(&rmi_dev->dev,
- + "Failed to disable irq for wake: %d\n",
- + retval);
- + }
-
- - /*
- - * Call rmi_process_interrupt_requests() after enabling irq,
- - * otherwise we may lose interrupt on edge-triggered systems.
- - */
- - irq_flags = irq_get_trigger_type(pdata->irq);
- - if (irq_flags & IRQ_TYPE_EDGE_BOTH)
- - rmi_process_interrupt_requests(rmi_dev);
- + /*
- + * Call rmi_process_interrupt_requests() after enabling irq,
- + * otherwise we may lose interrupt on edge-triggered systems.
- + */
- + irq_flags = irq_get_trigger_type(pdata->irq);
- + if (irq_flags & IRQ_TYPE_EDGE_BOTH)
- + rmi_process_interrupt_requests(rmi_dev);
- + } else {
- + data->enabled = true;
- + }
-
- out:
- mutex_unlock(&data->enabled_mutex);
- @@ -922,20 +936,22 @@ void rmi_disable_irq(struct rmi_device *rmi_dev, bool enable_wake)
- goto out;
-
- data->enabled = false;
- - disable_irq(irq);
- - if (enable_wake && device_may_wakeup(rmi_dev->xport->dev)) {
- - retval = enable_irq_wake(irq);
- - if (retval)
- - dev_warn(&rmi_dev->dev,
- - "Failed to enable irq for wake: %d\n",
- - retval);
- - }
- -
- - /* make sure the fifo is clean */
- - while (!kfifo_is_empty(&data->attn_fifo)) {
- - count = kfifo_get(&data->attn_fifo, &attn_data);
- - if (count)
- - kfree(attn_data.data);
- + if (irq) {
- + disable_irq(irq);
- + if (enable_wake && device_may_wakeup(rmi_dev->xport->dev)) {
- + retval = enable_irq_wake(irq);
- + if (retval)
- + dev_warn(&rmi_dev->dev,
- + "Failed to enable irq for wake: %d\n",
- + retval);
- + }
- + } else {
- + /* make sure the fifo is clean */
- + while (!kfifo_is_empty(&data->attn_fifo)) {
- + count = kfifo_get(&data->attn_fifo, &attn_data);
- + if (count)
- + kfree(attn_data.data);
- + }
- }
-
- out:
- @@ -981,6 +997,8 @@ static int rmi_driver_remove(struct device *dev)
- irq_domain_remove(data->irqdomain);
- data->irqdomain = NULL;
-
- + cancel_work_sync(&data->attn_work);
- +
- rmi_f34_remove_sysfs(rmi_dev);
- rmi_free_function_list(rmi_dev);
-
- @@ -1219,9 +1237,15 @@ static int rmi_driver_probe(struct device *dev)
- }
- }
-
- - retval = rmi_irq_init(rmi_dev);
- - if (retval < 0)
- - goto err_destroy_functions;
- + if (pdata->irq) {
- + retval = rmi_irq_init(rmi_dev);
- + if (retval < 0)
- + goto err_destroy_functions;
- + }
- +
- + data->enabled = true;
- +
- + INIT_WORK(&data->attn_work, attn_callback);
-
- if (data->f01_container->dev.driver) {
- /* Driver already bound, so enable ATTN now. */
- diff --git a/drivers/iommu/iommu.c b/drivers/iommu/iommu.c
- index 3a67e636287a..eb5e796277d6 100644
- --- a/drivers/iommu/iommu.c
- +++ b/drivers/iommu/iommu.c
- @@ -8,6 +8,7 @@
-
- #include <linux/amba/bus.h>
- #include <linux/device.h>
- +#include <linux/dmi.h>
- #include <linux/kernel.h>
- #include <linux/bits.h>
- #include <linux/bug.h>
- @@ -2931,6 +2932,27 @@ int iommu_dev_disable_feature(struct device *dev, enum iommu_dev_features feat)
- }
- EXPORT_SYMBOL_GPL(iommu_dev_disable_feature);
-
- +#ifdef CONFIG_ARM64
- +static int __init iommu_quirks(void)
- +{
- + const char *vendor, *name;
- +
- + vendor = dmi_get_system_info(DMI_SYS_VENDOR);
- + name = dmi_get_system_info(DMI_PRODUCT_NAME);
- +
- + if (vendor &&
- + (strncmp(vendor, "GIGABYTE", 8) == 0 && name &&
- + (strncmp(name, "R120", 4) == 0 ||
- + strncmp(name, "R270", 4) == 0))) {
- + pr_warn("Gigabyte %s detected, force iommu passthrough mode", name);
- + iommu_def_domain_type = IOMMU_DOMAIN_IDENTITY;
- + }
- +
- + return 0;
- +}
- +arch_initcall(iommu_quirks);
- +#endif
- +
- /**
- * iommu_setup_default_domain - Set the default_domain for the group
- * @group: Group to change
- diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c
- index b3976dcb71f1..54b122be6f35 100644
- --- a/drivers/pci/quirks.c
- +++ b/drivers/pci/quirks.c
- @@ -4426,6 +4426,30 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_BROADCOM, 0x9000,
- DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_BROADCOM, 0x9084,
- quirk_bridge_cavm_thrx2_pcie_root);
-
- +/*
- + * PCI BAR 5 is not setup correctly for the on-board AHCI controller
- + * on Broadcom's Vulcan processor. Added a quirk to fix BAR 5 by
- + * using BAR 4's resources which are populated correctly and NOT
- + * actually used by the AHCI controller.
- + */
- +static void quirk_fix_vulcan_ahci_bars(struct pci_dev *dev)
- +{
- + struct resource *r = &dev->resource[4];
- +
- + if (!(r->flags & IORESOURCE_MEM) || (r->start == 0))
- + return;
- +
- + /* Set BAR5 resource to BAR4 */
- + dev->resource[5] = *r;
- +
- + /* Update BAR5 in pci config space */
- + pci_write_config_dword(dev, PCI_BASE_ADDRESS_5, r->start);
- +
- + /* Clear BAR4's resource */
- + memset(r, 0, sizeof(*r));
- +}
- +DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_BROADCOM, 0x9027, quirk_fix_vulcan_ahci_bars);
- +
- /*
- * Intersil/Techwell TW686[4589]-based video capture cards have an empty (zero)
- * class code. Fix it.
- diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
- index e80c33cdad2b..3064ada8b6b8 100644
- --- a/drivers/scsi/sd.c
- +++ b/drivers/scsi/sd.c
- @@ -118,6 +118,14 @@ static const char *sd_cache_types[] = {
- "write back, no read (daft)"
- };
-
- +static const char *sd_probe_types[] = { "async", "sync" };
- +
- +static char sd_probe_type[6] = "async";
- +module_param_string(probe, sd_probe_type, sizeof(sd_probe_type),
- + S_IRUGO|S_IWUSR);
- +MODULE_PARM_DESC(probe, "async or sync. Setting to 'sync' disables asynchronous "
- + "device number assignments (sda, sdb, ...).");
- +
- static void sd_set_flush_flag(struct scsi_disk *sdkp)
- {
- bool wc = false, fua = false;
- @@ -4084,6 +4092,8 @@ static int __init init_sd(void)
- goto err_out_class;
- }
-
- + if (!strcmp(sd_probe_type, "sync"))
- + sd_template.gendrv.probe_type = PROBE_FORCE_SYNCHRONOUS;
- err = scsi_register_driver(&sd_template.gendrv);
- if (err)
- goto err_out_driver;
- diff --git a/include/linux/efi.h b/include/linux/efi.h
- index 80b21d1c6eaf..b66c0683f2fc 100644
- --- a/include/linux/efi.h
- +++ b/include/linux/efi.h
- @@ -44,6 +44,8 @@ struct screen_info;
- #define EFI_ABORTED (21 | (1UL << (BITS_PER_LONG-1)))
- #define EFI_SECURITY_VIOLATION (26 | (1UL << (BITS_PER_LONG-1)))
-
- +#define EFI_IS_ERROR(x) ((x) & (1UL << (BITS_PER_LONG-1)))
- +
- typedef unsigned long efi_status_t;
- typedef u8 efi_bool_t;
- typedef u16 efi_char16_t; /* UNICODE character */
- @@ -871,6 +873,14 @@ extern int __init efi_setup_pcdp_console(char *);
- #define EFI_MEM_ATTR 10 /* Did firmware publish an EFI_MEMORY_ATTRIBUTES table? */
- #define EFI_MEM_NO_SOFT_RESERVE 11 /* Is the kernel configured to ignore soft reservations? */
- #define EFI_PRESERVE_BS_REGIONS 12 /* Are EFI boot-services memory segments available? */
- +#define EFI_SECURE_BOOT 13 /* Are we in Secure Boot mode? */
- +
- +enum efi_secureboot_mode {
- + efi_secureboot_mode_unset,
- + efi_secureboot_mode_unknown,
- + efi_secureboot_mode_disabled,
- + efi_secureboot_mode_enabled,
- +};
-
- #ifdef CONFIG_EFI
- /*
- @@ -882,6 +892,8 @@ static inline bool efi_enabled(int feature)
- }
- extern void efi_reboot(enum reboot_mode reboot_mode, const char *__unused);
-
- +extern void __init efi_set_secure_boot(enum efi_secureboot_mode mode);
- +
- bool __pure __efi_soft_reserve_enabled(void);
-
- static inline bool __pure efi_soft_reserve_enabled(void)
- @@ -903,6 +915,8 @@ static inline bool efi_enabled(int feature)
- static inline void
- efi_reboot(enum reboot_mode reboot_mode, const char *__unused) {}
-
- +static inline void efi_set_secure_boot(enum efi_secureboot_mode mode) {}
- +
- static inline bool efi_soft_reserve_enabled(void)
- {
- return false;
- @@ -917,6 +931,7 @@ static inline void efi_find_mirror(void) {}
- #endif
-
- extern int efi_status_to_err(efi_status_t status);
- +extern const char *efi_status_to_str(efi_status_t status);
-
- /*
- * Variable Attributes
- @@ -1133,13 +1148,6 @@ static inline bool efi_runtime_disabled(void) { return true; }
- extern void efi_call_virt_check_flags(unsigned long flags, const void *caller);
- extern unsigned long efi_call_virt_save_flags(void);
-
- -enum efi_secureboot_mode {
- - efi_secureboot_mode_unset,
- - efi_secureboot_mode_unknown,
- - efi_secureboot_mode_disabled,
- - efi_secureboot_mode_enabled,
- -};
- -
- static inline
- enum efi_secureboot_mode efi_get_secureboot_mode(efi_get_variable_t *get_var)
- {
- diff --git a/include/linux/lsm_hook_defs.h b/include/linux/lsm_hook_defs.h
- index 489362b0cd85..49933ea3afa9 100644
- --- a/include/linux/lsm_hook_defs.h
- +++ b/include/linux/lsm_hook_defs.h
- @@ -407,6 +407,8 @@ LSM_HOOK(void, LSM_RET_VOID, bpf_prog_free_security, struct bpf_prog_aux *aux)
- #endif /* CONFIG_BPF_SYSCALL */
-
- LSM_HOOK(int, 0, locked_down, enum lockdown_reason what)
- +LSM_HOOK(int, 0, lock_kernel_down, const char *where, enum lockdown_reason level)
- +
-
- #ifdef CONFIG_PERF_EVENTS
- LSM_HOOK(int, 0, perf_event_open, struct perf_event_attr *attr, int type)
- diff --git a/include/linux/module.h b/include/linux/module.h
- index a98e188cf37b..2eef4246c2c9 100644
- --- a/include/linux/module.h
- +++ b/include/linux/module.h
- @@ -418,6 +418,7 @@ struct module {
- struct module_attribute *modinfo_attrs;
- const char *version;
- const char *srcversion;
- + const char *rhelversion;
- struct kobject *holders_dir;
-
- /* Exported symbols */
- diff --git a/include/linux/rh_kabi.h b/include/linux/rh_kabi.h
- new file mode 100644
- index 000000000000..e0d3353802bb
- --- /dev/null
- +++ b/include/linux/rh_kabi.h
- @@ -0,0 +1,172 @@
- +/*
- + * rh_kabi.h - Red Hat kABI abstraction header
- + *
- + * Copyright (c) 2014 Don Zickus
- + * Copyright (c) 2015-2017 Jiri Benc
- + * Copyright (c) 2015 Sabrina Dubroca, Hannes Frederic Sowa
- + * Copyright (c) 2016-2018 Prarit Bhargava
- + * Copyright (c) 2017 Paolo Abeni, Larry Woodman
- + *
- + * This file is released under the GPLv2.
- + * See the file COPYING for more details.
- + *
- + * These kabi macros hide the changes from the kabi checker and from the
- + * process that computes the exported symbols' checksums.
- + * They have 2 variants: one (defined under __GENKSYMS__) used when
- + * generating the checksums, and the other used when building the kernel's
- + * binaries.
- + *
- + * The use of these macros does not guarantee that the usage and modification
- + * of code is correct. As with all Red Hat only changes, an engineer must
- + * explain why the use of the macro is valid in the patch containing the
- + * changes.
- + *
- + */
- +
- +#ifndef _LINUX_RH_KABI_H
- +#define _LINUX_RH_KABI_H
- +
- +#include <linux/compiler.h>
- +#include <linux/stringify.h>
- +
- +/*
- + * RH_KABI_CONST
- + * Adds a new const modifier to a function parameter preserving the old
- + * checksum.
- + *
- + * RH_KABI_DEPRECATE
- + * Mark the element as deprecated and make it unusable by modules while
- + * preserving kABI checksums.
- + *
- + * RH_KABI_DEPRECATE_FN
- + * Mark the function pointer as deprecated and make it unusable by modules
- + * while preserving kABI checksums.
- + *
- + * RH_KABI_EXTEND
- + * Simple macro for adding a new element to a struct.
- + *
- + * Warning: only use if a hole exists for _all_ arches. Use pahole to verify.
- + *
- + * RH_KABI_FILL_HOLE
- + * Simple macro for filling a hole in a struct.
- + *
- + * RH_KABI_RENAME
- + * Simple macro for renaming an element without changing its type. This
- + * macro can be used in bitfields, for example.
- + *
- + * NOTE: does not include the final ';'
- + *
- + * RH_KABI_REPLACE
- + * Simple replacement of _orig with a union of _orig and _new.
- + *
- + * The RH_KABI_REPLACE* macros attempt to add the ability to use the '_new'
- + * element while preserving size alignment with the '_orig' element.
- + *
- + * The #ifdef __GENKSYMS__ preserves the kABI agreement, while the anonymous
- + * union structure preserves the size alignment (assuming the '_new' element
- + * is not bigger than the '_orig' element).
- + *
- + * RH_KABI_REPLACE_UNSAFE
- + * Unsafe version of RH_KABI_REPLACE. Only use for typedefs.
- + *
- + * RH_KABI_FORCE_CHANGE
- + * Force change of the symbol checksum. The argument of the macro is a
- + * version for cases we need to do this more than once.
- + *
- + * This macro does the opposite: it changes the symbol checksum without
- + * actually changing anything about the exported symbol. It is useful for
- + * symbols that are not whitelisted, we're changing them in an
- + * incompatible way and want to prevent 3rd party modules to silently
- + * corrupt memory. Instead, by changing the symbol checksum, such modules
- + * won't be loaded by the kernel. This macro should only be used as a
- + * last resort when all other KABI workarounds have failed.
- + *
- + * NOTE
- + * Don't use ';' after these macros as it messes up the kABI checker by
- + * changing what the resulting token string looks like. Instead let this
- + * macro add the ';' so it can be properly hidden from the kABI checker
- + * (mainly for RH_KABI_EXTEND, but applied to all macros for uniformity).
- + *
- + */
- +#ifdef __GENKSYMS__
- +
- +# define RH_KABI_CONST
- +# define RH_KABI_EXTEND(_new)
- +# define RH_KABI_FILL_HOLE(_new)
- +# define RH_KABI_FORCE_CHANGE(ver) __attribute__((rh_kabi_change ## ver))
- +# define RH_KABI_RENAME(_orig, _new) _orig
- +
- +# define _RH_KABI_DEPRECATE(_type, _orig) _type _orig
- +# define _RH_KABI_DEPRECATE_FN(_type, _orig, _args...) _type (*_orig)(_args)
- +# define _RH_KABI_REPLACE(_orig, _new) _orig
- +# define _RH_KABI_REPLACE_UNSAFE(_orig, _new) _orig
- +
- +#else
- +
- +# define RH_KABI_ALIGN_WARNING ". Disable CONFIG_RH_KABI_SIZE_ALIGN_CHECKS if debugging."
- +
- +# define RH_KABI_CONST const
- +# define RH_KABI_EXTEND(_new) _new;
- +# define RH_KABI_FILL_HOLE(_new) _new;
- +# define RH_KABI_FORCE_CHANGE(ver)
- +# define RH_KABI_RENAME(_orig, _new) _new
- +
- +
- +#if IS_BUILTIN(CONFIG_RH_KABI_SIZE_ALIGN_CHECKS)
- +# define __RH_KABI_CHECK_SIZE_ALIGN(_orig, _new) \
- + union { \
- + _Static_assert(sizeof(struct{_new;}) <= sizeof(struct{_orig;}), \
- + __FILE__ ":" __stringify(__LINE__) ": " __stringify(_new) " is larger than " __stringify(_orig) RH_KABI_ALIGN_WARNING); \
- + _Static_assert(__alignof__(struct{_new;}) <= __alignof__(struct{_orig;}), \
- + __FILE__ ":" __stringify(__LINE__) ": " __stringify(_orig) " is not aligned the same as " __stringify(_new) RH_KABI_ALIGN_WARNING); \
- + }
- +#else
- +# define __RH_KABI_CHECK_SIZE_ALIGN(_orig, _new)
- +#endif
- +
- +# define _RH_KABI_DEPRECATE(_type, _orig) _type rh_reserved_##_orig
- +# define _RH_KABI_DEPRECATE_FN(_type, _orig, _args...) \
- + _type (* rh_reserved_##_orig)(_args)
- +# define _RH_KABI_REPLACE(_orig, _new) \
- + union { \
- + _new; \
- + struct { \
- + _orig; \
- + } __UNIQUE_ID(rh_kabi_hide); \
- + __RH_KABI_CHECK_SIZE_ALIGN(_orig, _new); \
- + }
- +# define _RH_KABI_REPLACE_UNSAFE(_orig, _new) _new
- +
- +#endif /* __GENKSYMS__ */
- +
- +/* semicolon added wrappers for the RH_KABI_REPLACE macros */
- +# define RH_KABI_DEPRECATE(_type, _orig) _RH_KABI_DEPRECATE(_type, _orig);
- +# define RH_KABI_DEPRECATE_FN(_type, _orig, _args...) \
- + _RH_KABI_DEPRECATE_FN(_type, _orig, _args);
- +# define RH_KABI_REPLACE(_orig, _new) _RH_KABI_REPLACE(_orig, _new);
- +# define RH_KABI_REPLACE_UNSAFE(_orig, _new) _RH_KABI_REPLACE_UNSAFE(_orig, _new);
- +/*
- + * Macro for breaking up a random element into two smaller chunks using an
- + * anonymous struct inside an anonymous union.
- + */
- +# define RH_KABI_REPLACE2(orig, _new1, _new2) RH_KABI_REPLACE(orig, struct{ _new1; _new2;})
- +
- +# define RH_KABI_RESERVE(n) _RH_KABI_RESERVE(n);
- +/*
- + * Simple wrappers to replace standard Red Hat reserved elements.
- + */
- +# define RH_KABI_USE(n, _new) RH_KABI_REPLACE(_RH_KABI_RESERVE(n), _new)
- +/*
- + * Macros for breaking up a reserved element into two smaller chunks using
- + * an anonymous struct inside an anonymous union.
- + */
- +# define RH_KABI_USE2(n, _new1, _new2) RH_KABI_REPLACE(_RH_KABI_RESERVE(n), struct{ _new1; _new2; })
- +
- +/*
- + * We tried to standardize on Red Hat reserved names. These wrappers
- + * leverage those common names making it easier to read and find in the
- + * code.
- + */
- +# define _RH_KABI_RESERVE(n) unsigned long rh_reserved##n
- +
- +#endif /* _LINUX_RH_KABI_H */
- diff --git a/include/linux/rmi.h b/include/linux/rmi.h
- index ab7eea01ab42..fff7c5f737fc 100644
- --- a/include/linux/rmi.h
- +++ b/include/linux/rmi.h
- @@ -364,6 +364,7 @@ struct rmi_driver_data {
-
- struct rmi4_attn_data attn_data;
- DECLARE_KFIFO(attn_fifo, struct rmi4_attn_data, 16);
- + struct work_struct attn_work;
- };
-
- int rmi_register_transport_device(struct rmi_transport_dev *xport);
- diff --git a/include/linux/security.h b/include/linux/security.h
- index b50c7f566189..8a12e0f0a0fe 100644
- --- a/include/linux/security.h
- +++ b/include/linux/security.h
- @@ -486,6 +486,7 @@ int security_inode_notifysecctx(struct inode *inode, void *ctx, u32 ctxlen);
- int security_inode_setsecctx(struct dentry *dentry, void *ctx, u32 ctxlen);
- int security_inode_getsecctx(struct inode *inode, void **ctx, u32 *ctxlen);
- int security_locked_down(enum lockdown_reason what);
- +int security_lock_kernel_down(const char *where, enum lockdown_reason level);
- #else /* CONFIG_SECURITY */
-
- static inline int call_blocking_lsm_notifier(enum lsm_event event, void *data)
- @@ -1404,6 +1405,10 @@ static inline int security_locked_down(enum lockdown_reason what)
- {
- return 0;
- }
- +static inline int security_lock_kernel_down(const char *where, enum lockdown_reason level)
- +{
- + return 0;
- +}
- #endif /* CONFIG_SECURITY */
-
- #if defined(CONFIG_SECURITY) && defined(CONFIG_WATCH_QUEUE)
- diff --git a/kernel/module/main.c b/kernel/module/main.c
- index 34d9e718c2c7..1c8b40b19014 100644
- --- a/kernel/module/main.c
- +++ b/kernel/module/main.c
- @@ -528,6 +528,7 @@ static struct module_attribute modinfo_##field = { \
-
- MODINFO_ATTR(version);
- MODINFO_ATTR(srcversion);
- +MODINFO_ATTR(rhelversion);
-
- static struct {
- char name[MODULE_NAME_LEN + 1];
- @@ -980,6 +981,7 @@ struct module_attribute *modinfo_attrs[] = {
- &module_uevent,
- &modinfo_version,
- &modinfo_srcversion,
- + &modinfo_rhelversion,
- &modinfo_initstate,
- &modinfo_coresize,
- #ifdef CONFIG_ARCH_WANTS_MODULES_DATA_IN_VMALLOC
- diff --git a/kernel/module/signing.c b/kernel/module/signing.c
- index a2ff4242e623..f0d2be1ee4f1 100644
- --- a/kernel/module/signing.c
- +++ b/kernel/module/signing.c
- @@ -61,10 +61,17 @@ int mod_verify_sig(const void *mod, struct load_info *info)
- modlen -= sig_len + sizeof(ms);
- info->len = modlen;
-
- - return verify_pkcs7_signature(mod, modlen, mod + modlen, sig_len,
- + ret = verify_pkcs7_signature(mod, modlen, mod + modlen, sig_len,
- VERIFY_USE_SECONDARY_KEYRING,
- VERIFYING_MODULE_SIGNATURE,
- NULL, NULL);
- + if (ret == -ENOKEY && IS_ENABLED(CONFIG_INTEGRITY_PLATFORM_KEYRING)) {
- + ret = verify_pkcs7_signature(mod, modlen, mod + modlen, sig_len,
- + VERIFY_USE_PLATFORM_KEYRING,
- + VERIFYING_MODULE_SIGNATURE,
- + NULL, NULL);
- + }
- + return ret;
- }
-
- int module_sig_check(struct load_info *info, int flags)
- diff --git a/scripts/tags.sh b/scripts/tags.sh
- index a70d43723146..56d06b04f752 100755
- --- a/scripts/tags.sh
- +++ b/scripts/tags.sh
- @@ -16,6 +16,8 @@ fi
- ignore="$(echo "$RCS_FIND_IGNORE" | sed 's|\\||g' )"
- # tags and cscope files should also ignore MODVERSION *.mod.c files
- ignore="$ignore ( -name *.mod.c ) -prune -o"
- +# RHEL tags and cscope should also ignore redhat/rpm
- +ignore="$ignore ( -path redhat/rpm ) -prune -o"
-
- # ignore arbitrary directories
- if [ -n "${IGNORE_DIRS}" ]; then
- diff --git a/security/integrity/platform_certs/load_uefi.c b/security/integrity/platform_certs/load_uefi.c
- index d1fdd113450a..182e8090cfe8 100644
- --- a/security/integrity/platform_certs/load_uefi.c
- +++ b/security/integrity/platform_certs/load_uefi.c
- @@ -74,7 +74,8 @@ static __init void *get_cert_list(efi_char16_t *name, efi_guid_t *guid,
- return NULL;
-
- if (*status != EFI_BUFFER_TOO_SMALL) {
- - pr_err("Couldn't get size: 0x%lx\n", *status);
- + pr_err("Couldn't get size: %s (0x%lx)\n",
- + efi_status_to_str(*status), *status);
- return NULL;
- }
-
- @@ -85,7 +86,8 @@ static __init void *get_cert_list(efi_char16_t *name, efi_guid_t *guid,
- *status = efi.get_variable(name, guid, NULL, &lsize, db);
- if (*status != EFI_SUCCESS) {
- kfree(db);
- - pr_err("Error reading db var: 0x%lx\n", *status);
- + pr_err("Error reading db var: %s (0x%lx)\n",
- + efi_status_to_str(*status), *status);
- return NULL;
- }
-
- diff --git a/security/lockdown/Kconfig b/security/lockdown/Kconfig
- index e84ddf484010..d0501353a4b9 100644
- --- a/security/lockdown/Kconfig
- +++ b/security/lockdown/Kconfig
- @@ -16,6 +16,19 @@ config SECURITY_LOCKDOWN_LSM_EARLY
- subsystem is fully initialised. If enabled, lockdown will
- unconditionally be called before any other LSMs.
-
- +config LOCK_DOWN_IN_EFI_SECURE_BOOT
- + bool "Lock down the kernel in EFI Secure Boot mode"
- + default n
- + depends on EFI && SECURITY_LOCKDOWN_LSM_EARLY
- + help
- + UEFI Secure Boot provides a mechanism for ensuring that the firmware
- + will only load signed bootloaders and kernels. Secure boot mode may
- + be determined from EFI variables provided by the system firmware if
- + not indicated by the boot parameters.
- +
- + Enabling this option results in kernel lockdown being triggered if
- + EFI Secure Boot is set.
- +
- choice
- prompt "Kernel default lockdown mode"
- default LOCK_DOWN_KERNEL_FORCE_NONE
- diff --git a/security/lockdown/lockdown.c b/security/lockdown/lockdown.c
- index 68d19632aeb7..ef348935b6ff 100644
- --- a/security/lockdown/lockdown.c
- +++ b/security/lockdown/lockdown.c
- @@ -73,6 +73,7 @@ static int lockdown_is_locked_down(enum lockdown_reason what)
-
- static struct security_hook_list lockdown_hooks[] __ro_after_init = {
- LSM_HOOK_INIT(locked_down, lockdown_is_locked_down),
- + LSM_HOOK_INIT(lock_kernel_down, lock_kernel_down),
- };
-
- static int __init lockdown_lsm_init(void)
- diff --git a/security/security.c b/security/security.c
- index 407b51719f79..bffb48f02087 100644
- --- a/security/security.c
- +++ b/security/security.c
- @@ -5283,6 +5283,18 @@ int security_locked_down(enum lockdown_reason what)
- }
- EXPORT_SYMBOL(security_locked_down);
-
- +/**
- + * security_lock_kernel_down() - Put the kernel into lock-down mode.
- + *
- + * @where: Where the lock-down is originating from (e.g. command line option)
- + * @level: The lock-down level (can only increase)
- + */
- +int security_lock_kernel_down(const char *where, enum lockdown_reason level)
- +{
- + return call_int_hook(lock_kernel_down, 0, where, level);
- +}
- +EXPORT_SYMBOL(security_lock_kernel_down);
- +
- #ifdef CONFIG_PERF_EVENTS
- /**
- * security_perf_event_open() - Check if a perf event open is allowed
- diff --git a/sound/pci/hda/cs35l41_hda.c b/sound/pci/hda/cs35l41_hda.c
- index 3c157b006a5a..257f71e9ec07 100644
- --- a/sound/pci/hda/cs35l41_hda.c
- +++ b/sound/pci/hda/cs35l41_hda.c
- @@ -33,6 +33,9 @@
- #define CAL_AMBIENT_DSP_CTL_NAME "CAL_AMBIENT"
- #define CAL_DSP_CTL_TYPE 5
- #define CAL_DSP_CTL_ALG 205
- +#define CS35L41_UUID "50d90cdc-3de4-4f18-b528-c7fe3b71f40d"
- +#define CS35L41_DSM_GET_MUTE 5
- +#define CS35L41_NOTIFY_EVENT 0x91
-
- static bool firmware_autostart = 1;
- module_param(firmware_autostart, bool, 0444);
- @@ -563,6 +566,31 @@ static void cs35l41_hda_play_start(struct device *dev)
-
- }
-
- +static void cs35l41_mute(struct device *dev, bool mute)
- +{
- + struct cs35l41_hda *cs35l41 = dev_get_drvdata(dev);
- + struct regmap *reg = cs35l41->regmap;
- +
- + dev_dbg(dev, "Mute(%d:%d) Playback Started: %d\n", mute, cs35l41->mute_override,
- + cs35l41->playback_started);
- +
- + if (cs35l41->playback_started) {
- + if (mute || cs35l41->mute_override) {
- + dev_dbg(dev, "Muting\n");
- + regmap_multi_reg_write(reg, cs35l41_hda_mute, ARRAY_SIZE(cs35l41_hda_mute));
- + } else {
- + dev_dbg(dev, "Unmuting\n");
- + if (cs35l41->firmware_running) {
- + regmap_multi_reg_write(reg, cs35l41_hda_unmute_dsp,
- + ARRAY_SIZE(cs35l41_hda_unmute_dsp));
- + } else {
- + regmap_multi_reg_write(reg, cs35l41_hda_unmute,
- + ARRAY_SIZE(cs35l41_hda_unmute));
- + }
- + }
- + }
- +}
- +
- static void cs35l41_hda_play_done(struct device *dev)
- {
- struct cs35l41_hda *cs35l41 = dev_get_drvdata(dev);
- @@ -572,13 +600,7 @@ static void cs35l41_hda_play_done(struct device *dev)
-
- cs35l41_global_enable(dev, reg, cs35l41->hw_cfg.bst_type, 1,
- cs35l41->firmware_running);
- - if (cs35l41->firmware_running) {
- - regmap_multi_reg_write(reg, cs35l41_hda_unmute_dsp,
- - ARRAY_SIZE(cs35l41_hda_unmute_dsp));
- - } else {
- - regmap_multi_reg_write(reg, cs35l41_hda_unmute,
- - ARRAY_SIZE(cs35l41_hda_unmute));
- - }
- + cs35l41_mute(dev, false);
- }
-
- static void cs35l41_hda_pause_start(struct device *dev)
- @@ -588,7 +610,7 @@ static void cs35l41_hda_pause_start(struct device *dev)
-
- dev_dbg(dev, "Pause (Start)\n");
-
- - regmap_multi_reg_write(reg, cs35l41_hda_mute, ARRAY_SIZE(cs35l41_hda_mute));
- + cs35l41_mute(dev, true);
- cs35l41_global_enable(dev, reg, cs35l41->hw_cfg.bst_type, 0,
- cs35l41->firmware_running);
- }
- @@ -1116,6 +1138,53 @@ static int cs35l41_create_controls(struct cs35l41_hda *cs35l41)
- return 0;
- }
-
- +static bool cs35l41_dsm_supported(acpi_handle handle, unsigned int commands)
- +{
- + guid_t guid;
- +
- + guid_parse(CS35L41_UUID, &guid);
- +
- + return acpi_check_dsm(handle, &guid, 0, BIT(commands));
- +}
- +
- +static int cs35l41_get_acpi_mute_state(struct cs35l41_hda *cs35l41, acpi_handle handle)
- +{
- + guid_t guid;
- + union acpi_object *ret;
- + int mute = -ENODEV;
- +
- + guid_parse(CS35L41_UUID, &guid);
- +
- + if (cs35l41_dsm_supported(handle, CS35L41_DSM_GET_MUTE)) {
- + ret = acpi_evaluate_dsm(handle, &guid, 0, CS35L41_DSM_GET_MUTE, NULL);
- + mute = *ret->buffer.pointer;
- + dev_dbg(cs35l41->dev, "CS35L41_DSM_GET_MUTE: %d\n", mute);
- + }
- +
- + dev_dbg(cs35l41->dev, "%s: %d\n", __func__, mute);
- +
- + return mute;
- +}
- +
- +static void cs35l41_acpi_device_notify(acpi_handle handle, u32 event, struct device *dev)
- +{
- + struct cs35l41_hda *cs35l41 = dev_get_drvdata(dev);
- + int mute;
- +
- + if (event != CS35L41_NOTIFY_EVENT)
- + return;
- +
- + mute = cs35l41_get_acpi_mute_state(cs35l41, handle);
- + if (mute < 0) {
- + dev_warn(cs35l41->dev, "Unable to retrieve mute state: %d\n", mute);
- + return;
- + }
- +
- + dev_dbg(cs35l41->dev, "Requesting mute value: %d\n", mute);
- + cs35l41->mute_override = (mute > 0);
- + cs35l41_mute(cs35l41->dev, cs35l41->mute_override);
- +}
- +
- static int cs35l41_hda_bind(struct device *dev, struct device *master, void *master_data)
- {
- struct cs35l41_hda *cs35l41 = dev_get_drvdata(dev);
- @@ -1157,6 +1226,14 @@ static int cs35l41_hda_bind(struct device *dev, struct device *master, void *mas
- comps->playback_hook = cs35l41_hda_playback_hook;
- comps->pre_playback_hook = cs35l41_hda_pre_playback_hook;
- comps->post_playback_hook = cs35l41_hda_post_playback_hook;
- + comps->acpi_notify = cs35l41_acpi_device_notify;
- + comps->adev = cs35l41->dacpi;
- +
- + comps->acpi_notifications_supported = cs35l41_dsm_supported(acpi_device_handle(comps->adev),
- + CS35L41_DSM_GET_MUTE);
- +
- + cs35l41->mute_override = cs35l41_get_acpi_mute_state(cs35l41,
- + acpi_device_handle(cs35l41->dacpi)) > 0;
-
- mutex_unlock(&cs35l41->fw_mutex);
-
- @@ -1430,8 +1507,8 @@ static int cs35l41_hda_read_acpi(struct cs35l41_hda *cs35l41, const char *hid, i
- return -ENODEV;
- }
-
- + cs35l41->dacpi = adev;
- physdev = get_device(acpi_get_first_physical_node(adev));
- - acpi_dev_put(adev);
-
- sub = acpi_get_subsystem_id(ACPI_HANDLE(physdev));
- if (IS_ERR(sub))
- @@ -1541,6 +1618,7 @@ static int cs35l41_hda_read_acpi(struct cs35l41_hda *cs35l41, const char *hid, i
- hw_cfg->valid = false;
- hw_cfg->gpio1.valid = false;
- hw_cfg->gpio2.valid = false;
- + acpi_dev_put(cs35l41->dacpi);
- put_physdev:
- put_device(physdev);
-
- @@ -1644,10 +1722,7 @@ int cs35l41_hda_probe(struct device *dev, const char *device_name, int id, int i
- if (ret)
- goto err;
-
- - ret = regmap_multi_reg_write(cs35l41->regmap, cs35l41_hda_mute,
- - ARRAY_SIZE(cs35l41_hda_mute));
- - if (ret)
- - goto err;
- + cs35l41_mute(cs35l41->dev, true);
-
- INIT_WORK(&cs35l41->fw_load_work, cs35l41_fw_load_work);
- mutex_init(&cs35l41->fw_mutex);
- @@ -1684,6 +1759,8 @@ int cs35l41_hda_probe(struct device *dev, const char *device_name, int id, int i
- if (cs35l41_safe_reset(cs35l41->regmap, cs35l41->hw_cfg.bst_type))
- gpiod_set_value_cansleep(cs35l41->reset_gpio, 0);
- gpiod_put(cs35l41->reset_gpio);
- + gpiod_put(cs35l41->cs_gpio);
- + acpi_dev_put(cs35l41->dacpi);
- kfree(cs35l41->acpi_subsystem_id);
-
- return ret;
- @@ -1703,11 +1780,14 @@ void cs35l41_hda_remove(struct device *dev)
-
- component_del(cs35l41->dev, &cs35l41_hda_comp_ops);
-
- + acpi_dev_put(cs35l41->dacpi);
- +
- pm_runtime_put_noidle(cs35l41->dev);
-
- if (cs35l41_safe_reset(cs35l41->regmap, cs35l41->hw_cfg.bst_type))
- gpiod_set_value_cansleep(cs35l41->reset_gpio, 0);
- gpiod_put(cs35l41->reset_gpio);
- + gpiod_put(cs35l41->cs_gpio);
- kfree(cs35l41->acpi_subsystem_id);
- }
- EXPORT_SYMBOL_NS_GPL(cs35l41_hda_remove, SND_HDA_SCODEC_CS35L41);
- diff --git a/sound/pci/hda/cs35l41_hda.h b/sound/pci/hda/cs35l41_hda.h
- index b93bf762976e..3d925d677213 100644
- --- a/sound/pci/hda/cs35l41_hda.h
- +++ b/sound/pci/hda/cs35l41_hda.h
- @@ -10,6 +10,7 @@
- #ifndef __CS35L41_HDA_H__
- #define __CS35L41_HDA_H__
-
- +#include <linux/acpi.h>
- #include <linux/efi.h>
- #include <linux/regulator/consumer.h>
- #include <linux/gpio/consumer.h>
- @@ -34,8 +35,8 @@ struct cs35l41_amp_efi_data {
- } __packed;
-
- enum cs35l41_hda_spk_pos {
- - CS35l41_LEFT,
- - CS35l41_RIGHT,
- + CS35L41_LEFT,
- + CS35L41_RIGHT,
- };
-
- enum cs35l41_hda_gpio_function {
- @@ -49,6 +50,7 @@ struct cs35l41_hda {
- struct device *dev;
- struct regmap *regmap;
- struct gpio_desc *reset_gpio;
- + struct gpio_desc *cs_gpio;
- struct cs35l41_hw_cfg hw_cfg;
- struct hda_codec *codec;
-
- @@ -70,6 +72,8 @@ struct cs35l41_hda {
- bool halo_initialized;
- bool playback_started;
- struct cs_dsp cs_dsp;
- + struct acpi_device *dacpi;
- + bool mute_override;
- };
-
- enum halo_state {
- diff --git a/sound/pci/hda/cs35l41_hda_property.c b/sound/pci/hda/cs35l41_hda_property.c
- index b62a4e6968e2..c9eb70290973 100644
- --- a/sound/pci/hda/cs35l41_hda_property.c
- +++ b/sound/pci/hda/cs35l41_hda_property.c
- @@ -6,9 +6,300 @@
- //
- // Author: Stefan Binding <[email protected]>
-
- +#include <linux/acpi.h>
- #include <linux/gpio/consumer.h>
- #include <linux/string.h>
- #include "cs35l41_hda_property.h"
- +#include <linux/spi/spi.h>
- +
- +#define MAX_AMPS 4
- +
- +struct cs35l41_config {
- + const char *ssid;
- + enum {
- + SPI,
- + I2C
- + } bus;
- + int num_amps;
- + enum {
- + INTERNAL,
- + EXTERNAL
- + } boost_type;
- + u8 channel[MAX_AMPS];
- + int reset_gpio_index; /* -1 if no reset gpio */
- + int spkid_gpio_index; /* -1 if no spkid gpio */
- + int cs_gpio_index; /* -1 if no cs gpio, or cs-gpios already exists, max num amps == 2 */
- + int boost_ind_nanohenry; /* Required if boost_type == Internal */
- + int boost_peak_milliamp; /* Required if boost_type == Internal */
- + int boost_cap_microfarad; /* Required if boost_type == Internal */
- +};
- +
- +static const struct cs35l41_config cs35l41_config_table[] = {
- +/*
- + * Device 103C89C6 does have _DSD, however it is setup to use the wrong boost type.
- + * We can override the _DSD to correct the boost type here.
- + * Since this laptop has valid ACPI, we do not need to handle cs-gpios, since that already exists
- + * in the ACPI. The Reset GPIO is also valid, so we can use the Reset defined in _DSD.
- + */
- + { "103C89C6", SPI, 2, INTERNAL, { CS35L41_RIGHT, CS35L41_LEFT, 0, 0 }, -1, -1, -1, 1000, 4500, 24 },
- + { "104312AF", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431433", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431463", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431473", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, -1, 0, 1000, 4500, 24 },
- + { "10431483", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, -1, 0, 1000, 4500, 24 },
- + { "10431493", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "104314D3", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "104314E3", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431503", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431533", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431573", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431663", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, -1, 0, 1000, 4500, 24 },
- + { "104316D3", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 0, 0, 0 },
- + { "104316F3", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 0, 0, 0 },
- + { "104317F3", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431863", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "104318D3", I2C, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 0, 0, 0 },
- + { "10431C9F", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431CAF", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431CCF", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431CDF", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431CEF", SPI, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 1000, 4500, 24 },
- + { "10431D1F", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431DA2", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 0, 0, 0 },
- + { "10431E02", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 0, 0, 0 },
- + { "10431EE2", I2C, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, -1, -1, 0, 0, 0 },
- + { "10431F12", I2C, 2, INTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 0, 1, -1, 1000, 4500, 24 },
- + { "10431F1F", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, -1, 0, 0, 0, 0 },
- + { "10431F62", SPI, 2, EXTERNAL, { CS35L41_LEFT, CS35L41_RIGHT, 0, 0 }, 1, 2, 0, 0, 0, 0 },
- + {}
- +};
- +
- +static int cs35l41_add_gpios(struct cs35l41_hda *cs35l41, struct device *physdev, int reset_gpio,
- + int spkid_gpio, int cs_gpio_index, int num_amps)
- +{
- + struct acpi_gpio_mapping *gpio_mapping;
- + struct acpi_gpio_params *reset_gpio_params;
- + struct acpi_gpio_params *spkid_gpio_params;
- + struct acpi_gpio_params *cs_gpio_params;
- + unsigned int num_entries = 0;
- + unsigned int reset_index, spkid_index, csgpio_index;
- + int i;
- +
- + /*
- + * GPIO Mapping only needs to be done once, since it would be available for subsequent amps
- + */
- + if (cs35l41->dacpi->driver_gpios)
- + return 0;
- +
- + if (reset_gpio >= 0) {
- + reset_index = num_entries;
- + num_entries++;
- + }
- +
- + if (spkid_gpio >= 0) {
- + spkid_index = num_entries;
- + num_entries++;
- + }
- +
- + if ((cs_gpio_index >= 0) && (num_amps == 2)) {
- + csgpio_index = num_entries;
- + num_entries++;
- + }
- +
- + if (!num_entries)
- + return 0;
- +
- + /* must include termination entry */
- + num_entries++;
- +
- + gpio_mapping = devm_kcalloc(physdev, num_entries, sizeof(struct acpi_gpio_mapping),
- + GFP_KERNEL);
- +
- + if (!gpio_mapping)
- + goto err;
- +
- + if (reset_gpio >= 0) {
- + gpio_mapping[reset_index].name = "reset-gpios";
- + reset_gpio_params = devm_kcalloc(physdev, num_amps, sizeof(struct acpi_gpio_params),
- + GFP_KERNEL);
- + if (!reset_gpio_params)
- + goto err;
- +
- + for (i = 0; i < num_amps; i++)
- + reset_gpio_params[i].crs_entry_index = reset_gpio;
- +
- + gpio_mapping[reset_index].data = reset_gpio_params;
- + gpio_mapping[reset_index].size = num_amps;
- + }
- +
- + if (spkid_gpio >= 0) {
- + gpio_mapping[spkid_index].name = "spk-id-gpios";
- + spkid_gpio_params = devm_kcalloc(physdev, num_amps, sizeof(struct acpi_gpio_params),
- + GFP_KERNEL);
- + if (!spkid_gpio_params)
- + goto err;
- +
- + for (i = 0; i < num_amps; i++)
- + spkid_gpio_params[i].crs_entry_index = spkid_gpio;
- +
- + gpio_mapping[spkid_index].data = spkid_gpio_params;
- + gpio_mapping[spkid_index].size = num_amps;
- + }
- +
- + if ((cs_gpio_index >= 0) && (num_amps == 2)) {
- + gpio_mapping[csgpio_index].name = "cs-gpios";
- + /* only one GPIO CS is supported without using _DSD, obtained using index 0 */
- + cs_gpio_params = devm_kzalloc(physdev, sizeof(struct acpi_gpio_params), GFP_KERNEL);
- + if (!cs_gpio_params)
- + goto err;
- +
- + cs_gpio_params->crs_entry_index = cs_gpio_index;
- +
- + gpio_mapping[csgpio_index].data = cs_gpio_params;
- + gpio_mapping[csgpio_index].size = 1;
- + }
- +
- + return devm_acpi_dev_add_driver_gpios(physdev, gpio_mapping);
- +err:
- + devm_kfree(physdev, gpio_mapping);
- + devm_kfree(physdev, reset_gpio_params);
- + devm_kfree(physdev, spkid_gpio_params);
- + devm_kfree(physdev, cs_gpio_params);
- + return -ENOMEM;
- +}
- +
- +static int generic_dsd_config(struct cs35l41_hda *cs35l41, struct device *physdev, int id,
- + const char *hid)
- +{
- + struct cs35l41_hw_cfg *hw_cfg = &cs35l41->hw_cfg;
- + const struct cs35l41_config *cfg;
- + struct gpio_desc *cs_gpiod;
- + struct spi_device *spi;
- + bool dsd_found;
- + int ret;
- +
- + for (cfg = cs35l41_config_table; cfg->ssid; cfg++) {
- + if (!strcasecmp(cfg->ssid, cs35l41->acpi_subsystem_id))
- + break;
- + }
- +
- + if (!cfg->ssid)
- + return -ENOENT;
- +
- + if (!cs35l41->dacpi || cs35l41->dacpi != ACPI_COMPANION(physdev)) {
- + dev_err(cs35l41->dev, "ACPI Device does not match, cannot override _DSD.\n");
- + return -ENODEV;
- + }
- +
- + dev_info(cs35l41->dev, "Adding DSD properties for %s\n", cs35l41->acpi_subsystem_id);
- +
- + dsd_found = acpi_dev_has_props(cs35l41->dacpi);
- +
- + if (!dsd_found) {
- + ret = cs35l41_add_gpios(cs35l41, physdev, cfg->reset_gpio_index,
- + cfg->spkid_gpio_index, cfg->cs_gpio_index,
- + cfg->num_amps);
- + if (ret) {
- + dev_err(cs35l41->dev, "Error adding GPIO mapping: %d\n", ret);
- + return ret;
- + }
- + } else if (cfg->reset_gpio_index >= 0 || cfg->spkid_gpio_index >= 0) {
- + dev_warn(cs35l41->dev, "Cannot add Reset/Speaker ID/SPI CS GPIO Mapping, "
- + "_DSD already exists.\n");
- + }
- +
- + if (cfg->bus == SPI) {
- + cs35l41->index = id;
- + /*
- + * Manually set the Chip Select for the second amp <cs_gpio_index> in the node.
- + * This is only supported for systems with 2 amps, since we cannot expand the
- + * default number of chip selects without using cs-gpios
- + * The CS GPIO must be set high prior to communicating with the first amp (which
- + * uses a native chip select), to ensure the second amp does not clash with the
- + * first.
- + */
- + if (cfg->cs_gpio_index >= 0) {
- + spi = to_spi_device(cs35l41->dev);
- +
- + if (cfg->num_amps != 2) {
- + dev_warn(cs35l41->dev,
- + "Cannot update SPI CS, Number of Amps (%d) != 2\n",
- + cfg->num_amps);
- + } else if (dsd_found) {
- + dev_warn(cs35l41->dev,
- + "Cannot update SPI CS, _DSD already exists.\n");
- + } else {
- + /*
- + * This is obtained using driver_gpios, since only one GPIO for CS
- + * exists, this can be obtained using index 0.
- + */
- + cs_gpiod = gpiod_get_index(physdev, "cs", 0, GPIOD_OUT_LOW);
- + if (IS_ERR(cs_gpiod)) {
- + dev_err(cs35l41->dev,
- + "Unable to get Chip Select GPIO descriptor\n");
- + return PTR_ERR(cs_gpiod);
- + }
- + if (id == 1) {
- + spi_set_csgpiod(spi, 0, cs_gpiod);
- + cs35l41->cs_gpio = cs_gpiod;
- + } else {
- + gpiod_set_value_cansleep(cs_gpiod, true);
- + gpiod_put(cs_gpiod);
- + }
- + spi_setup(spi);
- + }
- + }
- + } else {
- + if (cfg->num_amps > 2)
- + /*
- + * i2c addresses for 3/4 amps are used in order: 0x40, 0x41, 0x42, 0x43,
- + * subtracting 0x40 would give zero-based index
- + */
- + cs35l41->index = id - 0x40;
- + else
- + /* i2c addr 0x40 for first amp (always), 0x41/0x42 for 2nd amp */
- + cs35l41->index = id == 0x40 ? 0 : 1;
- + }
- +
- + if (cfg->num_amps == 3)
- + /* 3 amps means a center channel, so no duplicate channels */
- + cs35l41->channel_index = 0;
- + else
- + /*
- + * if 4 amps, there are duplicate channels, so they need different indexes
- + * if 2 amps, no duplicate channels, channel_index would be 0
- + */
- + cs35l41->channel_index = cs35l41->index / 2;
- +
- + cs35l41->reset_gpio = fwnode_gpiod_get_index(acpi_fwnode_handle(cs35l41->dacpi), "reset",
- + cs35l41->index, GPIOD_OUT_LOW,
- + "cs35l41-reset");
- + cs35l41->speaker_id = cs35l41_get_speaker_id(physdev, cs35l41->index, cfg->num_amps, -1);
- +
- + hw_cfg->spk_pos = cfg->channel[cs35l41->index];
- +
- + if (cfg->boost_type == INTERNAL) {
- + hw_cfg->bst_type = CS35L41_INT_BOOST;
- + hw_cfg->bst_ind = cfg->boost_ind_nanohenry;
- + hw_cfg->bst_ipk = cfg->boost_peak_milliamp;
- + hw_cfg->bst_cap = cfg->boost_cap_microfarad;
- + hw_cfg->gpio1.func = CS35L41_NOT_USED;
- + hw_cfg->gpio1.valid = true;
- + } else {
- + hw_cfg->bst_type = CS35L41_EXT_BOOST;
- + hw_cfg->bst_ind = -1;
- + hw_cfg->bst_ipk = -1;
- + hw_cfg->bst_cap = -1;
- + hw_cfg->gpio1.func = CS35l41_VSPK_SWITCH;
- + hw_cfg->gpio1.valid = true;
- + }
- +
- + hw_cfg->gpio2.func = CS35L41_INTERRUPT;
- + hw_cfg->gpio2.valid = true;
- + hw_cfg->valid = true;
- +
- + return 0;
- +}
-
- /*
- * Device CLSA010(0/1) doesn't have _DSD so a gpiod_get by the label reset won't work.
- @@ -43,37 +334,6 @@ static int lenovo_legion_no_acpi(struct cs35l41_hda *cs35l41, struct device *phy
- return 0;
- }
-
- -/*
- - * Device 103C89C6 does have _DSD, however it is setup to use the wrong boost type.
- - * We can override the _DSD to correct the boost type here.
- - * Since this laptop has valid ACPI, we do not need to handle cs-gpios, since that already exists
- - * in the ACPI.
- - */
- -static int hp_vision_acpi_fix(struct cs35l41_hda *cs35l41, struct device *physdev, int id,
- - const char *hid)
- -{
- - struct cs35l41_hw_cfg *hw_cfg = &cs35l41->hw_cfg;
- -
- - dev_info(cs35l41->dev, "Adding DSD properties for %s\n", cs35l41->acpi_subsystem_id);
- -
- - cs35l41->index = id;
- - cs35l41->channel_index = 0;
- - cs35l41->reset_gpio = gpiod_get_index(physdev, NULL, 1, GPIOD_OUT_HIGH);
- - cs35l41->speaker_id = -ENOENT;
- - hw_cfg->spk_pos = cs35l41->index ? 1 : 0; // right:left
- - hw_cfg->gpio1.func = CS35L41_NOT_USED;
- - hw_cfg->gpio1.valid = true;
- - hw_cfg->gpio2.func = CS35L41_INTERRUPT;
- - hw_cfg->gpio2.valid = true;
- - hw_cfg->bst_type = CS35L41_INT_BOOST;
- - hw_cfg->bst_ind = 1000;
- - hw_cfg->bst_ipk = 4500;
- - hw_cfg->bst_cap = 24;
- - hw_cfg->valid = true;
- -
- - return 0;
- -}
- -
- struct cs35l41_prop_model {
- const char *hid;
- const char *ssid;
- @@ -84,7 +344,36 @@ struct cs35l41_prop_model {
- static const struct cs35l41_prop_model cs35l41_prop_model_table[] = {
- { "CLSA0100", NULL, lenovo_legion_no_acpi },
- { "CLSA0101", NULL, lenovo_legion_no_acpi },
- - { "CSC3551", "103C89C6", hp_vision_acpi_fix },
- + { "CSC3551", "103C89C6", generic_dsd_config },
- + { "CSC3551", "104312AF", generic_dsd_config },
- + { "CSC3551", "10431433", generic_dsd_config },
- + { "CSC3551", "10431463", generic_dsd_config },
- + { "CSC3551", "10431473", generic_dsd_config },
- + { "CSC3551", "10431483", generic_dsd_config },
- + { "CSC3551", "10431493", generic_dsd_config },
- + { "CSC3551", "104314D3", generic_dsd_config },
- + { "CSC3551", "104314E3", generic_dsd_config },
- + { "CSC3551", "10431503", generic_dsd_config },
- + { "CSC3551", "10431533", generic_dsd_config },
- + { "CSC3551", "10431573", generic_dsd_config },
- + { "CSC3551", "10431663", generic_dsd_config },
- + { "CSC3551", "104316D3", generic_dsd_config },
- + { "CSC3551", "104316F3", generic_dsd_config },
- + { "CSC3551", "104317F3", generic_dsd_config },
- + { "CSC3551", "10431863", generic_dsd_config },
- + { "CSC3551", "104318D3", generic_dsd_config },
- + { "CSC3551", "10431C9F", generic_dsd_config },
- + { "CSC3551", "10431CAF", generic_dsd_config },
- + { "CSC3551", "10431CCF", generic_dsd_config },
- + { "CSC3551", "10431CDF", generic_dsd_config },
- + { "CSC3551", "10431CEF", generic_dsd_config },
- + { "CSC3551", "10431D1F", generic_dsd_config },
- + { "CSC3551", "10431DA2", generic_dsd_config },
- + { "CSC3551", "10431E02", generic_dsd_config },
- + { "CSC3551", "10431EE2", generic_dsd_config },
- + { "CSC3551", "10431F12", generic_dsd_config },
- + { "CSC3551", "10431F1F", generic_dsd_config },
- + { "CSC3551", "10431F62", generic_dsd_config },
- {}
- };
-
- @@ -97,7 +386,7 @@ int cs35l41_add_dsd_properties(struct cs35l41_hda *cs35l41, struct device *physd
- if (!strcmp(model->hid, hid) &&
- (!model->ssid ||
- (cs35l41->acpi_subsystem_id &&
- - !strcmp(model->ssid, cs35l41->acpi_subsystem_id))))
- + !strcasecmp(model->ssid, cs35l41->acpi_subsystem_id))))
- return model->add_prop(cs35l41, physdev, id, hid);
- }
-
- diff --git a/sound/pci/hda/hda_component.h b/sound/pci/hda/hda_component.h
- index f170aec967c1..bbd6f0ed16c1 100644
- --- a/sound/pci/hda/hda_component.h
- +++ b/sound/pci/hda/hda_component.h
- @@ -6,6 +6,7 @@
- * Cirrus Logic International Semiconductor Ltd.
- */
-
- +#include <linux/acpi.h>
- #include <linux/component.h>
-
- #define HDA_MAX_COMPONENTS 4
- @@ -15,6 +16,9 @@ struct hda_component {
- struct device *dev;
- char name[HDA_MAX_NAME_SIZE];
- struct hda_codec *codec;
- + struct acpi_device *adev;
- + bool acpi_notifications_supported;
- + void (*acpi_notify)(acpi_handle handle, u32 event, struct device *dev);
- void (*pre_playback_hook)(struct device *dev, int action);
- void (*playback_hook)(struct device *dev, int action);
- void (*post_playback_hook)(struct device *dev, int action);
- diff --git a/sound/pci/hda/patch_realtek.c b/sound/pci/hda/patch_realtek.c
- index 0db9326b6f84..f747fcae24bb 100644
- --- a/sound/pci/hda/patch_realtek.c
- +++ b/sound/pci/hda/patch_realtek.c
- @@ -9981,22 +9981,28 @@ static const struct snd_pci_quirk alc269_fixup_tbl[] = {
- SND_PCI_QUIRK(0x1043, 0x1313, "Asus K42JZ", ALC269VB_FIXUP_ASUS_MIC_NO_PRESENCE),
- SND_PCI_QUIRK(0x1043, 0x13b0, "ASUS Z550SA", ALC256_FIXUP_ASUS_MIC),
- SND_PCI_QUIRK(0x1043, 0x1427, "Asus Zenbook UX31E", ALC269VB_FIXUP_ASUS_ZENBOOK),
- - SND_PCI_QUIRK(0x1043, 0x1433, "ASUS GX650P", ALC285_FIXUP_ASUS_I2C_HEADSET_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1463, "Asus GA402X", ALC285_FIXUP_ASUS_I2C_HEADSET_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1473, "ASUS GU604V", ALC285_FIXUP_ASUS_HEADSET_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1483, "ASUS GU603V", ALC285_FIXUP_ASUS_HEADSET_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1493, "ASUS GV601V", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1433, "ASUS GX650PY/PZ/PV/PU/PYV/PZV/PIV/PVV", ALC285_FIXUP_ASUS_I2C_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1463, "Asus GA402X/GA402N", ALC285_FIXUP_ASUS_I2C_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1473, "ASUS GU604VI/VC/VE/VG/VJ/VQ/VU/VV/VY/VZ", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1483, "ASUS GU603VQ/VU/VV/VJ/VI", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1493, "ASUS GV601VV/VU/VJ/VQ/VI", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x14d3, "ASUS G614JY/JZ/JG", ALC245_FIXUP_CS35L41_SPI_2),
- + SND_PCI_QUIRK(0x1043, 0x14e3, "ASUS G513PI/PU/PV", ALC287_FIXUP_CS35L41_I2C_2),
- + SND_PCI_QUIRK(0x1043, 0x1503, "ASUS G733PY/PZ/PZV/PYV", ALC287_FIXUP_CS35L41_I2C_2),
- SND_PCI_QUIRK(0x1043, 0x1517, "Asus Zenbook UX31A", ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A),
- - SND_PCI_QUIRK(0x1043, 0x1533, "ASUS GV302XA", ALC287_FIXUP_CS35L41_I2C_2),
- - SND_PCI_QUIRK(0x1043, 0x1573, "ASUS GZ301V", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1533, "ASUS GV302XA/XJ/XQ/XU/XV/XI", ALC287_FIXUP_CS35L41_I2C_2),
- + SND_PCI_QUIRK(0x1043, 0x1573, "ASUS GZ301VV/VQ/VU/VJ/VA/VC/VE/VVC/VQC/VUC/VJC/VEC/VCC", ALC285_FIXUP_ASUS_HEADSET_MIC),
- SND_PCI_QUIRK(0x1043, 0x1662, "ASUS GV301QH", ALC294_FIXUP_ASUS_DUAL_SPK),
- - SND_PCI_QUIRK(0x1043, 0x1663, "ASUS GU603ZV", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1663, "ASUS GU603ZI/ZJ/ZQ/ZU/ZV", ALC285_FIXUP_ASUS_HEADSET_MIC),
- SND_PCI_QUIRK(0x1043, 0x1683, "ASUS UM3402YAR", ALC287_FIXUP_CS35L41_I2C_2),
- SND_PCI_QUIRK(0x1043, 0x16b2, "ASUS GU603", ALC289_FIXUP_ASUS_GA401),
- + SND_PCI_QUIRK(0x1043, 0x16d3, "ASUS UX5304VA", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x16e3, "ASUS UX50", ALC269_FIXUP_STEREO_DMIC),
- + SND_PCI_QUIRK(0x1043, 0x16f3, "ASUS UX7602VI/BZ", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x1740, "ASUS UX430UA", ALC295_FIXUP_ASUS_DACS),
- SND_PCI_QUIRK(0x1043, 0x17d1, "ASUS UX431FL", ALC294_FIXUP_ASUS_DUAL_SPK),
- - SND_PCI_QUIRK(0x1043, 0x17f3, "ROG Ally RC71L_RC71L", ALC294_FIXUP_ASUS_ALLY),
- + SND_PCI_QUIRK(0x1043, 0x17f3, "ROG Ally NR2301L/X", ALC294_FIXUP_ASUS_ALLY),
- + SND_PCI_QUIRK(0x1043, 0x1863, "ASUS UX6404VI/VV", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x1881, "ASUS Zephyrus S/M", ALC294_FIXUP_ASUS_GX502_PINS),
- SND_PCI_QUIRK(0x1043, 0x18b1, "Asus MJ401TA", ALC256_FIXUP_ASUS_HEADSET_MIC),
- SND_PCI_QUIRK(0x1043, 0x18d3, "ASUS UM3504DA", ALC294_FIXUP_CS35L41_I2C_2),
- @@ -10021,10 +10027,13 @@ static const struct snd_pci_quirk alc269_fixup_tbl[] = {
- SND_PCI_QUIRK(0x1043, 0x1c43, "ASUS UX8406MA", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x1c62, "ASUS GU603", ALC289_FIXUP_ASUS_GA401),
- SND_PCI_QUIRK(0x1043, 0x1c92, "ASUS ROG Strix G15", ALC285_FIXUP_ASUS_G533Z_PINS),
- - SND_PCI_QUIRK(0x1043, 0x1c9f, "ASUS G614JI", ALC285_FIXUP_ASUS_HEADSET_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1caf, "ASUS G634JYR/JZR", ALC285_FIXUP_ASUS_SPI_REAR_SPEAKERS),
- + SND_PCI_QUIRK(0x1043, 0x1c9f, "ASUS G614JU/JV/JI", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1caf, "ASUS G634JY/JZ/JI/JG", ALC285_FIXUP_ASUS_SPI_REAR_SPEAKERS),
- SND_PCI_QUIRK(0x1043, 0x1ccd, "ASUS X555UB", ALC256_FIXUP_ASUS_MIC),
- - SND_PCI_QUIRK(0x1043, 0x1d1f, "ASUS ROG Strix G17 2023 (G713PV)", ALC287_FIXUP_CS35L41_I2C_2),
- + SND_PCI_QUIRK(0x1043, 0x1ccf, "ASUS G814JU/JV/JI", ALC245_FIXUP_CS35L41_SPI_2),
- + SND_PCI_QUIRK(0x1043, 0x1cdf, "ASUS G814JY/JZ/JG", ALC245_FIXUP_CS35L41_SPI_2),
- + SND_PCI_QUIRK(0x1043, 0x1cef, "ASUS G834JY/JZ/JI/JG", ALC285_FIXUP_ASUS_HEADSET_MIC),
- + SND_PCI_QUIRK(0x1043, 0x1d1f, "ASUS G713PI/PU/PV/PVN", ALC287_FIXUP_CS35L41_I2C_2),
- SND_PCI_QUIRK(0x1043, 0x1d42, "ASUS Zephyrus G14 2022", ALC289_FIXUP_ASUS_GA401),
- SND_PCI_QUIRK(0x1043, 0x1d4e, "ASUS TM420", ALC256_FIXUP_ASUS_HPE),
- SND_PCI_QUIRK(0x1043, 0x1da2, "ASUS UP6502ZA/ZD", ALC245_FIXUP_CS35L41_SPI_2),
- @@ -10040,6 +10049,7 @@ static const struct snd_pci_quirk alc269_fixup_tbl[] = {
- SND_PCI_QUIRK(0x1043, 0x1c52, "ASUS Zephyrus G15 2022", ALC289_FIXUP_ASUS_GA401),
- SND_PCI_QUIRK(0x1043, 0x1f11, "ASUS Zephyrus G14", ALC289_FIXUP_ASUS_GA401),
- SND_PCI_QUIRK(0x1043, 0x1f12, "ASUS UM5302", ALC287_FIXUP_CS35L41_I2C_2),
- + SND_PCI_QUIRK(0x1043, 0x1f1f, "ASUS H7604JI/JV/J3D", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x1f62, "ASUS UX7602ZM", ALC245_FIXUP_CS35L41_SPI_2),
- SND_PCI_QUIRK(0x1043, 0x1f92, "ASUS ROG Flow X16", ALC289_FIXUP_ASUS_GA401),
- SND_PCI_QUIRK(0x1043, 0x3030, "ASUS ZN270IE", ALC256_FIXUP_ASUS_AIO_GPIO2),
|