| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Copyright (C) 2023 Raspberry Pi Ltd.
- *
- * Clock driver for RP1 PCIe multifunction chip.
- */
- #include <linux/bitfield.h>
- #include <linux/clk-provider.h>
- #include <linux/regmap.h>
- #include <linux/math64.h>
- #include <linux/module.h>
- #include <linux/platform_device.h>
- #include <linux/units.h>
- #include <dt-bindings/clock/raspberrypi,rp1-clocks.h>
- #define PLL_SYS_OFFSET 0x08000
- #define PLL_SYS_CS (PLL_SYS_OFFSET + 0x00)
- #define PLL_SYS_PWR (PLL_SYS_OFFSET + 0x04)
- #define PLL_SYS_FBDIV_INT (PLL_SYS_OFFSET + 0x08)
- #define PLL_SYS_FBDIV_FRAC (PLL_SYS_OFFSET + 0x0c)
- #define PLL_SYS_PRIM (PLL_SYS_OFFSET + 0x10)
- #define PLL_SYS_SEC (PLL_SYS_OFFSET + 0x14)
- #define PLL_AUDIO_OFFSET 0x0c000
- #define PLL_AUDIO_CS (PLL_AUDIO_OFFSET + 0x00)
- #define PLL_AUDIO_PWR (PLL_AUDIO_OFFSET + 0x04)
- #define PLL_AUDIO_FBDIV_INT (PLL_AUDIO_OFFSET + 0x08)
- #define PLL_AUDIO_FBDIV_FRAC (PLL_AUDIO_OFFSET + 0x0c)
- #define PLL_AUDIO_PRIM (PLL_AUDIO_OFFSET + 0x10)
- #define PLL_AUDIO_SEC (PLL_AUDIO_OFFSET + 0x14)
- #define PLL_AUDIO_TERN (PLL_AUDIO_OFFSET + 0x18)
- #define PLL_VIDEO_OFFSET 0x10000
- #define PLL_VIDEO_CS (PLL_VIDEO_OFFSET + 0x00)
- #define PLL_VIDEO_PWR (PLL_VIDEO_OFFSET + 0x04)
- #define PLL_VIDEO_FBDIV_INT (PLL_VIDEO_OFFSET + 0x08)
- #define PLL_VIDEO_FBDIV_FRAC (PLL_VIDEO_OFFSET + 0x0c)
- #define PLL_VIDEO_PRIM (PLL_VIDEO_OFFSET + 0x10)
- #define PLL_VIDEO_SEC (PLL_VIDEO_OFFSET + 0x14)
- #define GPCLK_OE_CTRL 0x00000
- #define CLK_SYS_OFFSET 0x00014
- #define CLK_SYS_CTRL (CLK_SYS_OFFSET + 0x00)
- #define CLK_SYS_DIV_INT (CLK_SYS_OFFSET + 0x04)
- #define CLK_SYS_SEL (CLK_SYS_OFFSET + 0x0c)
- #define CLK_SLOW_OFFSET 0x00024
- #define CLK_SLOW_SYS_CTRL (CLK_SLOW_OFFSET + 0x00)
- #define CLK_SLOW_SYS_DIV_INT (CLK_SLOW_OFFSET + 0x04)
- #define CLK_SLOW_SYS_SEL (CLK_SLOW_OFFSET + 0x0c)
- #define CLK_DMA_OFFSET 0x00044
- #define CLK_DMA_CTRL (CLK_DMA_OFFSET + 0x00)
- #define CLK_DMA_DIV_INT (CLK_DMA_OFFSET + 0x04)
- #define CLK_DMA_SEL (CLK_DMA_OFFSET + 0x0c)
- #define CLK_UART_OFFSET 0x00054
- #define CLK_UART_CTRL (CLK_UART_OFFSET + 0x00)
- #define CLK_UART_DIV_INT (CLK_UART_OFFSET + 0x04)
- #define CLK_UART_SEL (CLK_UART_OFFSET + 0x0c)
- #define CLK_ETH_OFFSET 0x00064
- #define CLK_ETH_CTRL (CLK_ETH_OFFSET + 0x00)
- #define CLK_ETH_DIV_INT (CLK_ETH_OFFSET + 0x04)
- #define CLK_ETH_SEL (CLK_ETH_OFFSET + 0x0c)
- #define CLK_PWM0_OFFSET 0x00074
- #define CLK_PWM0_CTRL (CLK_PWM0_OFFSET + 0x00)
- #define CLK_PWM0_DIV_INT (CLK_PWM0_OFFSET + 0x04)
- #define CLK_PWM0_DIV_FRAC (CLK_PWM0_OFFSET + 0x08)
- #define CLK_PWM0_SEL (CLK_PWM0_OFFSET + 0x0c)
- #define CLK_PWM1_OFFSET 0x00084
- #define CLK_PWM1_CTRL (CLK_PWM1_OFFSET + 0x00)
- #define CLK_PWM1_DIV_INT (CLK_PWM1_OFFSET + 0x04)
- #define CLK_PWM1_DIV_FRAC (CLK_PWM1_OFFSET + 0x08)
- #define CLK_PWM1_SEL (CLK_PWM1_OFFSET + 0x0c)
- #define CLK_AUDIO_IN_OFFSET 0x00094
- #define CLK_AUDIO_IN_CTRL (CLK_AUDIO_IN_OFFSET + 0x00)
- #define CLK_AUDIO_IN_DIV_INT (CLK_AUDIO_IN_OFFSET + 0x04)
- #define CLK_AUDIO_IN_SEL (CLK_AUDIO_IN_OFFSET + 0x0c)
- #define CLK_AUDIO_OUT_OFFSET 0x000a4
- #define CLK_AUDIO_OUT_CTRL (CLK_AUDIO_OUT_OFFSET + 0x00)
- #define CLK_AUDIO_OUT_DIV_INT (CLK_AUDIO_OUT_OFFSET + 0x04)
- #define CLK_AUDIO_OUT_SEL (CLK_AUDIO_OUT_OFFSET + 0x0c)
- #define CLK_I2S_OFFSET 0x000b4
- #define CLK_I2S_CTRL (CLK_I2S_OFFSET + 0x00)
- #define CLK_I2S_DIV_INT (CLK_I2S_OFFSET + 0x04)
- #define CLK_I2S_SEL (CLK_I2S_OFFSET + 0x0c)
- #define CLK_MIPI0_CFG_OFFSET 0x000c4
- #define CLK_MIPI0_CFG_CTRL (CLK_MIPI0_CFG_OFFSET + 0x00)
- #define CLK_MIPI0_CFG_DIV_INT (CLK_MIPI0_CFG_OFFSET + 0x04)
- #define CLK_MIPI0_CFG_SEL (CLK_MIPI0_CFG_OFFSET + 0x0c)
- #define CLK_MIPI1_CFG_OFFSET 0x000d4
- #define CLK_MIPI1_CFG_CTRL (CLK_MIPI1_CFG_OFFSET + 0x00)
- #define CLK_MIPI1_CFG_DIV_INT (CLK_MIPI1_CFG_OFFSET + 0x04)
- #define CLK_MIPI1_CFG_SEL (CLK_MIPI1_CFG_OFFSET + 0x0c)
- #define CLK_PCIE_AUX_OFFSET 0x000e4
- #define CLK_PCIE_AUX_CTRL (CLK_PCIE_AUX_OFFSET + 0x00)
- #define CLK_PCIE_AUX_DIV_INT (CLK_PCIE_AUX_OFFSET + 0x04)
- #define CLK_PCIE_AUX_SEL (CLK_PCIE_AUX_OFFSET + 0x0c)
- #define CLK_USBH0_MICROFRAME_OFFSET 0x000f4
- #define CLK_USBH0_MICROFRAME_CTRL (CLK_USBH0_MICROFRAME_OFFSET + 0x00)
- #define CLK_USBH0_MICROFRAME_DIV_INT (CLK_USBH0_MICROFRAME_OFFSET + 0x04)
- #define CLK_USBH0_MICROFRAME_SEL (CLK_USBH0_MICROFRAME_OFFSET + 0x0c)
- #define CLK_USBH1_MICROFRAME_OFFSET 0x00104
- #define CLK_USBH1_MICROFRAME_CTRL (CLK_USBH1_MICROFRAME_OFFSET + 0x00)
- #define CLK_USBH1_MICROFRAME_DIV_INT (CLK_USBH1_MICROFRAME_OFFSET + 0x04)
- #define CLK_USBH1_MICROFRAME_SEL (CLK_USBH1_MICROFRAME_OFFSET + 0x0c)
- #define CLK_USBH0_SUSPEND_OFFSET 0x00114
- #define CLK_USBH0_SUSPEND_CTRL (CLK_USBH0_SUSPEND_OFFSET + 0x00)
- #define CLK_USBH0_SUSPEND_DIV_INT (CLK_USBH0_SUSPEND_OFFSET + 0x04)
- #define CLK_USBH0_SUSPEND_SEL (CLK_USBH0_SUSPEND_OFFSET + 0x0c)
- #define CLK_USBH1_SUSPEND_OFFSET 0x00124
- #define CLK_USBH1_SUSPEND_CTRL (CLK_USBH1_SUSPEND_OFFSET + 0x00)
- #define CLK_USBH1_SUSPEND_DIV_INT (CLK_USBH1_SUSPEND_OFFSET + 0x04)
- #define CLK_USBH1_SUSPEND_SEL (CLK_USBH1_SUSPEND_OFFSET + 0x0c)
- #define CLK_ETH_TSU_OFFSET 0x00134
- #define CLK_ETH_TSU_CTRL (CLK_ETH_TSU_OFFSET + 0x00)
- #define CLK_ETH_TSU_DIV_INT (CLK_ETH_TSU_OFFSET + 0x04)
- #define CLK_ETH_TSU_SEL (CLK_ETH_TSU_OFFSET + 0x0c)
- #define CLK_ADC_OFFSET 0x00144
- #define CLK_ADC_CTRL (CLK_ADC_OFFSET + 0x00)
- #define CLK_ADC_DIV_INT (CLK_ADC_OFFSET + 0x04)
- #define CLK_ADC_SEL (CLK_ADC_OFFSET + 0x0c)
- #define CLK_SDIO_TIMER_OFFSET 0x00154
- #define CLK_SDIO_TIMER_CTRL (CLK_SDIO_TIMER_OFFSET + 0x00)
- #define CLK_SDIO_TIMER_DIV_INT (CLK_SDIO_TIMER_OFFSET + 0x04)
- #define CLK_SDIO_TIMER_SEL (CLK_SDIO_TIMER_OFFSET + 0x0c)
- #define CLK_SDIO_ALT_SRC_OFFSET 0x00164
- #define CLK_SDIO_ALT_SRC_CTRL (CLK_SDIO_ALT_SRC_OFFSET + 0x00)
- #define CLK_SDIO_ALT_SRC_DIV_INT (CLK_SDIO_ALT_SRC_OFFSET + 0x04)
- #define CLK_SDIO_ALT_SRC_SEL (CLK_SDIO_ALT_SRC_OFFSET + 0x0c)
- #define CLK_GP0_OFFSET 0x00174
- #define CLK_GP0_CTRL (CLK_GP0_OFFSET + 0x00)
- #define CLK_GP0_DIV_INT (CLK_GP0_OFFSET + 0x04)
- #define CLK_GP0_DIV_FRAC (CLK_GP0_OFFSET + 0x08)
- #define CLK_GP0_SEL (CLK_GP0_OFFSET + 0x0c)
- #define CLK_GP1_OFFSET 0x00184
- #define CLK_GP1_CTRL (CLK_GP1_OFFSET + 0x00)
- #define CLK_GP1_DIV_INT (CLK_GP1_OFFSET + 0x04)
- #define CLK_GP1_DIV_FRAC (CLK_GP1_OFFSET + 0x08)
- #define CLK_GP1_SEL (CLK_GP1_OFFSET + 0x0c)
- #define CLK_GP2_OFFSET 0x00194
- #define CLK_GP2_CTRL (CLK_GP2_OFFSET + 0x00)
- #define CLK_GP2_DIV_INT (CLK_GP2_OFFSET + 0x04)
- #define CLK_GP2_DIV_FRAC (CLK_GP2_OFFSET + 0x08)
- #define CLK_GP2_SEL (CLK_GP2_OFFSET + 0x0c)
- #define CLK_GP3_OFFSET 0x001a4
- #define CLK_GP3_CTRL (CLK_GP3_OFFSET + 0x00)
- #define CLK_GP3_DIV_INT (CLK_GP3_OFFSET + 0x04)
- #define CLK_GP3_DIV_FRAC (CLK_GP3_OFFSET + 0x08)
- #define CLK_GP3_SEL (CLK_GP3_OFFSET + 0x0c)
- #define CLK_GP4_OFFSET 0x001b4
- #define CLK_GP4_CTRL (CLK_GP4_OFFSET + 0x00)
- #define CLK_GP4_DIV_INT (CLK_GP4_OFFSET + 0x04)
- #define CLK_GP4_DIV_FRAC (CLK_GP4_OFFSET + 0x08)
- #define CLK_GP4_SEL (CLK_GP4_OFFSET + 0x0c)
- #define CLK_GP5_OFFSET 0x001c4
- #define CLK_GP5_CTRL (CLK_GP5_OFFSET + 0x00)
- #define CLK_GP5_DIV_INT (CLK_GP5_OFFSET + 0x04)
- #define CLK_GP5_DIV_FRAC (CLK_GP5_OFFSET + 0x08)
- #define CLK_GP5_SEL (CLK_GP5_OFFSET + 0x0c)
- #define CLK_SYS_RESUS_CTRL 0x0020c
- #define CLK_SLOW_SYS_RESUS_CTRL 0x00214
- #define FC0_OFFSET 0x0021c
- #define FC0_REF_KHZ (FC0_OFFSET + 0x00)
- #define FC0_MIN_KHZ (FC0_OFFSET + 0x04)
- #define FC0_MAX_KHZ (FC0_OFFSET + 0x08)
- #define FC0_DELAY (FC0_OFFSET + 0x0c)
- #define FC0_INTERVAL (FC0_OFFSET + 0x10)
- #define FC0_SRC (FC0_OFFSET + 0x14)
- #define FC0_STATUS (FC0_OFFSET + 0x18)
- #define FC0_RESULT (FC0_OFFSET + 0x1c)
- #define FC_SIZE 0x20
- #define FC_COUNT 8
- #define FC_NUM(idx, off) ((idx) * 32 + (off))
- #define AUX_SEL 1
- #define VIDEO_CLOCKS_OFFSET 0x4000
- #define VIDEO_CLK_VEC_CTRL (VIDEO_CLOCKS_OFFSET + 0x0000)
- #define VIDEO_CLK_VEC_DIV_INT (VIDEO_CLOCKS_OFFSET + 0x0004)
- #define VIDEO_CLK_VEC_SEL (VIDEO_CLOCKS_OFFSET + 0x000c)
- #define VIDEO_CLK_DPI_CTRL (VIDEO_CLOCKS_OFFSET + 0x0010)
- #define VIDEO_CLK_DPI_DIV_INT (VIDEO_CLOCKS_OFFSET + 0x0014)
- #define VIDEO_CLK_DPI_SEL (VIDEO_CLOCKS_OFFSET + 0x001c)
- #define VIDEO_CLK_MIPI0_DPI_CTRL (VIDEO_CLOCKS_OFFSET + 0x0020)
- #define VIDEO_CLK_MIPI0_DPI_DIV_INT (VIDEO_CLOCKS_OFFSET + 0x0024)
- #define VIDEO_CLK_MIPI0_DPI_DIV_FRAC (VIDEO_CLOCKS_OFFSET + 0x0028)
- #define VIDEO_CLK_MIPI0_DPI_SEL (VIDEO_CLOCKS_OFFSET + 0x002c)
- #define VIDEO_CLK_MIPI1_DPI_CTRL (VIDEO_CLOCKS_OFFSET + 0x0030)
- #define VIDEO_CLK_MIPI1_DPI_DIV_INT (VIDEO_CLOCKS_OFFSET + 0x0034)
- #define VIDEO_CLK_MIPI1_DPI_DIV_FRAC (VIDEO_CLOCKS_OFFSET + 0x0038)
- #define VIDEO_CLK_MIPI1_DPI_SEL (VIDEO_CLOCKS_OFFSET + 0x003c)
- #define DIV_INT_8BIT_MAX GENMASK(7, 0) /* max divide for most clocks */
- #define DIV_INT_16BIT_MAX GENMASK(15, 0) /* max divide for GPx, PWM */
- #define DIV_INT_24BIT_MAX GENMASK(23, 0) /* max divide for CLK_SYS */
- #define FC0_STATUS_DONE BIT(4)
- #define FC0_STATUS_RUNNING BIT(8)
- #define FC0_RESULT_FRAC_SHIFT 5
- #define PLL_PRIM_DIV1_MASK GENMASK(18, 16)
- #define PLL_PRIM_DIV2_MASK GENMASK(14, 12)
- #define PLL_SEC_DIV_MASK GENMASK(12, 8)
- #define PLL_CS_LOCK BIT(31)
- #define PLL_CS_REFDIV_MASK BIT(1)
- #define PLL_PWR_PD BIT(0)
- #define PLL_PWR_DACPD BIT(1)
- #define PLL_PWR_DSMPD BIT(2)
- #define PLL_PWR_POSTDIVPD BIT(3)
- #define PLL_PWR_4PHASEPD BIT(4)
- #define PLL_PWR_VCOPD BIT(5)
- #define PLL_PWR_MASK GENMASK(5, 0)
- #define PLL_SEC_RST BIT(16)
- #define PLL_SEC_IMPL BIT(31)
- /* PLL phase output for both PRI and SEC */
- #define PLL_PH_EN BIT(4)
- #define PLL_PH_PHASE_SHIFT 0
- #define RP1_PLL_PHASE_0 0
- #define RP1_PLL_PHASE_90 1
- #define RP1_PLL_PHASE_180 2
- #define RP1_PLL_PHASE_270 3
- /* Clock fields for all clocks */
- #define CLK_CTRL_ENABLE BIT(11)
- #define CLK_CTRL_AUXSRC_MASK GENMASK(9, 5)
- #define CLK_CTRL_SRC_SHIFT 0
- #define CLK_DIV_FRAC_BITS 16
- #define LOCK_TIMEOUT_US 100000
- #define LOCK_POLL_DELAY_US 5
- #define MAX_CLK_PARENTS 16
- #define PLL_DIV_INVALID 19
- /*
- * Secondary PLL channel output divider table.
- * Divider values range from 8 to 19, where
- * 19 means invalid.
- */
- static const struct clk_div_table pll_sec_div_table[] = {
- { 0x00, PLL_DIV_INVALID },
- { 0x01, PLL_DIV_INVALID },
- { 0x02, PLL_DIV_INVALID },
- { 0x03, PLL_DIV_INVALID },
- { 0x04, PLL_DIV_INVALID },
- { 0x05, PLL_DIV_INVALID },
- { 0x06, PLL_DIV_INVALID },
- { 0x07, PLL_DIV_INVALID },
- { 0x08, 8 },
- { 0x09, 9 },
- { 0x0a, 10 },
- { 0x0b, 11 },
- { 0x0c, 12 },
- { 0x0d, 13 },
- { 0x0e, 14 },
- { 0x0f, 15 },
- { 0x10, 16 },
- { 0x11, 17 },
- { 0x12, 18 },
- { 0x13, PLL_DIV_INVALID },
- { 0x14, PLL_DIV_INVALID },
- { 0x15, PLL_DIV_INVALID },
- { 0x16, PLL_DIV_INVALID },
- { 0x17, PLL_DIV_INVALID },
- { 0x18, PLL_DIV_INVALID },
- { 0x19, PLL_DIV_INVALID },
- { 0x1a, PLL_DIV_INVALID },
- { 0x1b, PLL_DIV_INVALID },
- { 0x1c, PLL_DIV_INVALID },
- { 0x1d, PLL_DIV_INVALID },
- { 0x1e, PLL_DIV_INVALID },
- { 0x1f, PLL_DIV_INVALID },
- { 0 }
- };
- struct rp1_clockman {
- struct device *dev;
- void __iomem *regs;
- struct regmap *regmap;
- spinlock_t regs_lock; /* spinlock for all clocks */
- /* Must be last */
- struct clk_hw_onecell_data onecell;
- };
- struct rp1_pll_core_data {
- u32 cs_reg;
- u32 pwr_reg;
- u32 fbdiv_int_reg;
- u32 fbdiv_frac_reg;
- u32 fc0_src;
- };
- struct rp1_pll_data {
- u32 ctrl_reg;
- u32 fc0_src;
- };
- struct rp1_pll_ph_data {
- unsigned int phase;
- unsigned int fixed_divider;
- u32 ph_reg;
- u32 fc0_src;
- };
- struct rp1_pll_divider_data {
- u32 sec_reg;
- u32 fc0_src;
- };
- struct rp1_clock_data {
- int num_std_parents;
- int num_aux_parents;
- u32 oe_mask;
- u32 clk_src_mask;
- u32 ctrl_reg;
- u32 div_int_reg;
- u32 div_frac_reg;
- u32 sel_reg;
- u32 div_int_max;
- unsigned long max_freq;
- u32 fc0_src;
- };
- struct rp1_clk_desc {
- struct clk_hw *(*clk_register)(struct rp1_clockman *clockman,
- struct rp1_clk_desc *desc);
- const void *data;
- struct clk_hw hw;
- struct rp1_clockman *clockman;
- unsigned long cached_rate;
- struct clk_divider div;
- };
- static struct rp1_clk_desc *clk_audio_core;
- static struct rp1_clk_desc *clk_audio;
- static struct rp1_clk_desc *clk_i2s;
- static struct clk_hw *clk_xosc;
- static inline
- void clockman_write(struct rp1_clockman *clockman, u32 reg, u32 val)
- {
- regmap_write(clockman->regmap, reg, val);
- }
- static inline u32 clockman_read(struct rp1_clockman *clockman, u32 reg)
- {
- u32 val;
- regmap_read(clockman->regmap, reg, &val);
- return val;
- }
- static int rp1_pll_core_is_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_core = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_core->clockman;
- const struct rp1_pll_core_data *data = pll_core->data;
- u32 pwr = clockman_read(clockman, data->pwr_reg);
- return (pwr & PLL_PWR_PD) || (pwr & PLL_PWR_POSTDIVPD);
- }
- static int rp1_pll_core_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_core = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_core->clockman;
- const struct rp1_pll_core_data *data = pll_core->data;
- u32 fbdiv_frac, val;
- int ret;
- spin_lock(&clockman->regs_lock);
- if (!(clockman_read(clockman, data->cs_reg) & PLL_CS_LOCK)) {
- /* Reset to a known state. */
- clockman_write(clockman, data->pwr_reg, PLL_PWR_MASK);
- clockman_write(clockman, data->fbdiv_int_reg, 20);
- clockman_write(clockman, data->fbdiv_frac_reg, 0);
- clockman_write(clockman, data->cs_reg, PLL_CS_REFDIV_MASK);
- }
- /* Come out of reset. */
- fbdiv_frac = clockman_read(clockman, data->fbdiv_frac_reg);
- clockman_write(clockman, data->pwr_reg, fbdiv_frac ? 0 : PLL_PWR_DSMPD);
- spin_unlock(&clockman->regs_lock);
- /* Wait for the PLL to lock. */
- ret = regmap_read_poll_timeout(clockman->regmap, data->cs_reg, val,
- val & PLL_CS_LOCK,
- LOCK_POLL_DELAY_US, LOCK_TIMEOUT_US);
- if (ret)
- dev_err(clockman->dev, "%s: can't lock PLL\n",
- clk_hw_get_name(hw));
- return ret;
- }
- static void rp1_pll_core_off(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_core = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_core->clockman;
- const struct rp1_pll_core_data *data = pll_core->data;
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->pwr_reg, 0);
- spin_unlock(&clockman->regs_lock);
- }
- static inline unsigned long get_pll_core_divider(struct clk_hw *hw,
- unsigned long rate,
- unsigned long parent_rate,
- u32 *div_int, u32 *div_frac)
- {
- u32 fbdiv_int, fbdiv_frac;
- unsigned long calc_rate;
- u64 shifted_fbdiv_int;
- u64 div_fp64; /* 32.32 fixed point fraction. */
- /* Factor of reference clock to VCO frequency. */
- div_fp64 = (u64)(rate) << 32;
- div_fp64 = DIV_ROUND_CLOSEST_ULL(div_fp64, parent_rate);
- /* Round the fractional component at 24 bits. */
- div_fp64 += 1 << (32 - 24 - 1);
- fbdiv_int = div_fp64 >> 32;
- fbdiv_frac = (div_fp64 >> (32 - 24)) & 0xffffff;
- shifted_fbdiv_int = (u64)fbdiv_int << 24;
- calc_rate = (u64)parent_rate * (shifted_fbdiv_int + fbdiv_frac);
- calc_rate += BIT(23);
- calc_rate >>= 24;
- *div_int = fbdiv_int;
- *div_frac = fbdiv_frac;
- return calc_rate;
- }
- static int rp1_pll_core_set_rate(struct clk_hw *hw,
- unsigned long rate, unsigned long parent_rate)
- {
- struct rp1_clk_desc *pll_core = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_core->clockman;
- const struct rp1_pll_core_data *data = pll_core->data;
- u32 fbdiv_int, fbdiv_frac;
- /* Disable dividers to start with. */
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->fbdiv_int_reg, 0);
- clockman_write(clockman, data->fbdiv_frac_reg, 0);
- spin_unlock(&clockman->regs_lock);
- get_pll_core_divider(hw, rate, parent_rate,
- &fbdiv_int, &fbdiv_frac);
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->pwr_reg, fbdiv_frac ? 0 : PLL_PWR_DSMPD);
- clockman_write(clockman, data->fbdiv_int_reg, fbdiv_int);
- clockman_write(clockman, data->fbdiv_frac_reg, fbdiv_frac);
- spin_unlock(&clockman->regs_lock);
- /* Check that reference frequency is no greater than VCO / 16. */
- if (WARN_ON_ONCE(parent_rate > (rate / 16)))
- return -ERANGE;
- spin_lock(&clockman->regs_lock);
- /* Don't need to divide ref unless parent_rate > (output freq / 16) */
- clockman_write(clockman, data->cs_reg,
- clockman_read(clockman, data->cs_reg) |
- PLL_CS_REFDIV_MASK);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static unsigned long rp1_pll_core_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *pll_core = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_core->clockman;
- const struct rp1_pll_core_data *data = pll_core->data;
- u32 fbdiv_int, fbdiv_frac;
- unsigned long calc_rate;
- u64 shifted_fbdiv_int;
- fbdiv_int = clockman_read(clockman, data->fbdiv_int_reg);
- fbdiv_frac = clockman_read(clockman, data->fbdiv_frac_reg);
- shifted_fbdiv_int = (u64)fbdiv_int << 24;
- calc_rate = (u64)parent_rate * (shifted_fbdiv_int + fbdiv_frac);
- calc_rate += BIT(23);
- calc_rate >>= 24;
- return calc_rate;
- }
- static int rp1_pll_core_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- u32 fbdiv_int, fbdiv_frac;
- req->rate = get_pll_core_divider(hw, req->rate, req->best_parent_rate,
- &fbdiv_int,
- &fbdiv_frac);
- return 0;
- }
- static void get_pll_prim_dividers(unsigned long rate, unsigned long parent_rate,
- u32 *divider1, u32 *divider2)
- {
- unsigned int div1, div2;
- unsigned int best_div1 = 7, best_div2 = 7;
- unsigned long best_rate_diff =
- abs_diff(DIV_ROUND_CLOSEST(parent_rate, best_div1 * best_div2), rate);
- unsigned long rate_diff, calc_rate;
- for (div1 = 1; div1 <= 7; div1++) {
- for (div2 = 1; div2 <= div1; div2++) {
- calc_rate = DIV_ROUND_CLOSEST(parent_rate, div1 * div2);
- rate_diff = abs_diff(calc_rate, rate);
- if (calc_rate == rate) {
- best_div1 = div1;
- best_div2 = div2;
- goto done;
- } else if (rate_diff < best_rate_diff) {
- best_div1 = div1;
- best_div2 = div2;
- best_rate_diff = rate_diff;
- }
- }
- }
- done:
- *divider1 = best_div1;
- *divider2 = best_div2;
- }
- static int rp1_pll_set_rate(struct clk_hw *hw,
- unsigned long rate, unsigned long parent_rate)
- {
- struct rp1_clk_desc *pll = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll->clockman;
- const struct rp1_pll_data *data = pll->data;
- u32 prim, prim_div1, prim_div2;
- get_pll_prim_dividers(rate, parent_rate, &prim_div1, &prim_div2);
- spin_lock(&clockman->regs_lock);
- prim = clockman_read(clockman, data->ctrl_reg);
- prim &= ~PLL_PRIM_DIV1_MASK;
- prim |= FIELD_PREP(PLL_PRIM_DIV1_MASK, prim_div1);
- prim &= ~PLL_PRIM_DIV2_MASK;
- prim |= FIELD_PREP(PLL_PRIM_DIV2_MASK, prim_div2);
- clockman_write(clockman, data->ctrl_reg, prim);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static unsigned long rp1_pll_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *pll = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll->clockman;
- const struct rp1_pll_data *data = pll->data;
- u32 prim, prim_div1, prim_div2;
- prim = clockman_read(clockman, data->ctrl_reg);
- prim_div1 = FIELD_GET(PLL_PRIM_DIV1_MASK, prim);
- prim_div2 = FIELD_GET(PLL_PRIM_DIV2_MASK, prim);
- if (!prim_div1 || !prim_div2) {
- dev_err(clockman->dev, "%s: (%s) zero divider value\n",
- __func__, clk_hw_get_name(hw));
- return 0;
- }
- return DIV_ROUND_CLOSEST(parent_rate, prim_div1 * prim_div2);
- }
- static int rp1_pll_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- struct clk_hw *clk_audio_hw = &clk_audio->hw;
- u32 div1, div2;
- if (hw == clk_audio_hw && clk_audio->cached_rate == req->rate)
- req->best_parent_rate = clk_audio_core->cached_rate;
- get_pll_prim_dividers(req->rate, req->best_parent_rate, &div1, &div2);
- req->rate = DIV_ROUND_CLOSEST(req->best_parent_rate, div1 * div2);
- return 0;
- }
- static int rp1_pll_ph_is_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_ph = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_ph->clockman;
- const struct rp1_pll_ph_data *data = pll_ph->data;
- return !!(clockman_read(clockman, data->ph_reg) & PLL_PH_EN);
- }
- static int rp1_pll_ph_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_ph = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_ph->clockman;
- const struct rp1_pll_ph_data *data = pll_ph->data;
- u32 ph_reg;
- spin_lock(&clockman->regs_lock);
- ph_reg = clockman_read(clockman, data->ph_reg);
- ph_reg |= data->phase << PLL_PH_PHASE_SHIFT;
- ph_reg |= PLL_PH_EN;
- clockman_write(clockman, data->ph_reg, ph_reg);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static void rp1_pll_ph_off(struct clk_hw *hw)
- {
- struct rp1_clk_desc *pll_ph = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = pll_ph->clockman;
- const struct rp1_pll_ph_data *data = pll_ph->data;
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->ph_reg,
- clockman_read(clockman, data->ph_reg) & ~PLL_PH_EN);
- spin_unlock(&clockman->regs_lock);
- }
- static unsigned long rp1_pll_ph_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *pll_ph = container_of(hw, struct rp1_clk_desc, hw);
- const struct rp1_pll_ph_data *data = pll_ph->data;
- return parent_rate / data->fixed_divider;
- }
- static int rp1_pll_ph_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- struct rp1_clk_desc *pll_ph = container_of(hw, struct rp1_clk_desc, hw);
- const struct rp1_pll_ph_data *data = pll_ph->data;
- req->rate = req->best_parent_rate / data->fixed_divider;
- return 0;
- }
- static int rp1_pll_divider_is_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *divider = container_of(hw, struct rp1_clk_desc, div.hw);
- struct rp1_clockman *clockman = divider->clockman;
- const struct rp1_pll_data *data = divider->data;
- return !(clockman_read(clockman, data->ctrl_reg) & PLL_SEC_RST);
- }
- static int rp1_pll_divider_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *divider = container_of(hw, struct rp1_clk_desc, div.hw);
- struct rp1_clockman *clockman = divider->clockman;
- const struct rp1_pll_data *data = divider->data;
- spin_lock(&clockman->regs_lock);
- /* Check the implementation bit is set! */
- WARN_ON(!(clockman_read(clockman, data->ctrl_reg) & PLL_SEC_IMPL));
- clockman_write(clockman, data->ctrl_reg,
- clockman_read(clockman, data->ctrl_reg) & ~PLL_SEC_RST);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static void rp1_pll_divider_off(struct clk_hw *hw)
- {
- struct rp1_clk_desc *divider = container_of(hw, struct rp1_clk_desc, div.hw);
- struct rp1_clockman *clockman = divider->clockman;
- const struct rp1_pll_data *data = divider->data;
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->ctrl_reg,
- clockman_read(clockman, data->ctrl_reg) | PLL_SEC_RST);
- spin_unlock(&clockman->regs_lock);
- }
- static int rp1_pll_divider_set_rate(struct clk_hw *hw,
- unsigned long rate,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *divider = container_of(hw, struct rp1_clk_desc, div.hw);
- struct rp1_clockman *clockman = divider->clockman;
- const struct rp1_pll_data *data = divider->data;
- u32 div, sec;
- div = DIV_ROUND_UP_ULL(parent_rate, rate);
- div = clamp(div, 8u, 19u);
- spin_lock(&clockman->regs_lock);
- sec = clockman_read(clockman, data->ctrl_reg);
- sec &= ~PLL_SEC_DIV_MASK;
- sec |= FIELD_PREP(PLL_SEC_DIV_MASK, div);
- /* Must keep the divider in reset to change the value. */
- sec |= PLL_SEC_RST;
- clockman_write(clockman, data->ctrl_reg, sec);
- /* must sleep 10 pll vco cycles */
- ndelay(div64_ul(10ULL * div * NSEC_PER_SEC, parent_rate));
- sec &= ~PLL_SEC_RST;
- clockman_write(clockman, data->ctrl_reg, sec);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static unsigned long rp1_pll_divider_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- return clk_divider_ops.recalc_rate(hw, parent_rate);
- }
- static int rp1_pll_divider_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- req->rate = clk_divider_ops.determine_rate(hw, req);
- return 0;
- }
- static int rp1_clock_is_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- return !!(clockman_read(clockman, data->ctrl_reg) & CLK_CTRL_ENABLE);
- }
- static unsigned long rp1_clock_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- u64 calc_rate;
- u64 div;
- u32 frac;
- div = clockman_read(clockman, data->div_int_reg);
- frac = (data->div_frac_reg != 0) ?
- clockman_read(clockman, data->div_frac_reg) : 0;
- /* If the integer portion of the divider is 0, treat it as 2^16 */
- if (!div)
- div = 1 << 16;
- div = (div << CLK_DIV_FRAC_BITS) | (frac >> (32 - CLK_DIV_FRAC_BITS));
- calc_rate = (u64)parent_rate << CLK_DIV_FRAC_BITS;
- calc_rate = div64_u64(calc_rate, div);
- return calc_rate;
- }
- static int rp1_clock_on(struct clk_hw *hw)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->ctrl_reg,
- clockman_read(clockman, data->ctrl_reg) | CLK_CTRL_ENABLE);
- /* If this is a GPCLK, turn on the output-enable */
- if (data->oe_mask)
- clockman_write(clockman, GPCLK_OE_CTRL,
- clockman_read(clockman, GPCLK_OE_CTRL) | data->oe_mask);
- spin_unlock(&clockman->regs_lock);
- return 0;
- }
- static void rp1_clock_off(struct clk_hw *hw)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->ctrl_reg,
- clockman_read(clockman, data->ctrl_reg) & ~CLK_CTRL_ENABLE);
- /* If this is a GPCLK, turn off the output-enable */
- if (data->oe_mask)
- clockman_write(clockman, GPCLK_OE_CTRL,
- clockman_read(clockman, GPCLK_OE_CTRL) & ~data->oe_mask);
- spin_unlock(&clockman->regs_lock);
- }
- static u32 rp1_clock_choose_div(unsigned long rate, unsigned long parent_rate,
- const struct rp1_clock_data *data)
- {
- u64 div;
- /*
- * Due to earlier rounding, calculated parent_rate may differ from
- * expected value. Don't fail on a small discrepancy near unity divide.
- */
- if (!rate || rate > parent_rate + (parent_rate >> CLK_DIV_FRAC_BITS))
- return 0;
- /*
- * Always express div in fixed-point format for fractional division;
- * If no fractional divider is present, the fraction part will be zero.
- */
- if (data->div_frac_reg) {
- div = (u64)parent_rate << CLK_DIV_FRAC_BITS;
- div = DIV_ROUND_CLOSEST_ULL(div, rate);
- } else {
- div = DIV_ROUND_CLOSEST_ULL(parent_rate, rate);
- div <<= CLK_DIV_FRAC_BITS;
- }
- div = clamp(div,
- 1ull << CLK_DIV_FRAC_BITS,
- (u64)data->div_int_max << CLK_DIV_FRAC_BITS);
- return div;
- }
- static u8 rp1_clock_get_parent(struct clk_hw *hw)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- u32 sel, ctrl;
- u8 parent;
- /* Sel is one-hot, so find the first bit set */
- sel = clockman_read(clockman, data->sel_reg);
- parent = ffs(sel) - 1;
- /* sel == 0 implies the parent clock is not enabled yet. */
- if (!sel) {
- /* Read the clock src from the CTRL register instead */
- ctrl = clockman_read(clockman, data->ctrl_reg);
- parent = (ctrl & data->clk_src_mask) >> CLK_CTRL_SRC_SHIFT;
- }
- if (parent >= data->num_std_parents)
- parent = AUX_SEL;
- if (parent == AUX_SEL) {
- /*
- * Clock parent is an auxiliary source, so get the parent from
- * the AUXSRC register field.
- */
- ctrl = clockman_read(clockman, data->ctrl_reg);
- parent = FIELD_GET(CLK_CTRL_AUXSRC_MASK, ctrl);
- parent += data->num_std_parents;
- }
- return parent;
- }
- static int rp1_clock_set_parent(struct clk_hw *hw, u8 index)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- u32 ctrl, sel;
- spin_lock(&clockman->regs_lock);
- ctrl = clockman_read(clockman, data->ctrl_reg);
- if (index >= data->num_std_parents) {
- /* This is an aux source request */
- if (index >= data->num_std_parents + data->num_aux_parents) {
- spin_unlock(&clockman->regs_lock);
- return -EINVAL;
- }
- /* Select parent from aux list */
- ctrl &= ~CLK_CTRL_AUXSRC_MASK;
- ctrl |= FIELD_PREP(CLK_CTRL_AUXSRC_MASK, index - data->num_std_parents);
- /* Set src to aux list */
- ctrl &= ~data->clk_src_mask;
- ctrl |= (AUX_SEL << CLK_CTRL_SRC_SHIFT) & data->clk_src_mask;
- } else {
- ctrl &= ~data->clk_src_mask;
- ctrl |= (index << CLK_CTRL_SRC_SHIFT) & data->clk_src_mask;
- }
- clockman_write(clockman, data->ctrl_reg, ctrl);
- spin_unlock(&clockman->regs_lock);
- sel = rp1_clock_get_parent(hw);
- if (sel != index)
- return -EINVAL;
- return 0;
- }
- static int rp1_clock_set_rate_and_parent(struct clk_hw *hw,
- unsigned long rate,
- unsigned long parent_rate,
- u8 parent)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- struct rp1_clockman *clockman = clock->clockman;
- const struct rp1_clock_data *data = clock->data;
- u32 div = rp1_clock_choose_div(rate, parent_rate, data);
- spin_lock(&clockman->regs_lock);
- clockman_write(clockman, data->div_int_reg, div >> CLK_DIV_FRAC_BITS);
- if (data->div_frac_reg)
- clockman_write(clockman, data->div_frac_reg, div << (32 - CLK_DIV_FRAC_BITS));
- spin_unlock(&clockman->regs_lock);
- if (parent != 0xff)
- return rp1_clock_set_parent(hw, parent);
- return 0;
- }
- static int rp1_clock_set_rate(struct clk_hw *hw, unsigned long rate,
- unsigned long parent_rate)
- {
- return rp1_clock_set_rate_and_parent(hw, rate, parent_rate, 0xff);
- }
- static unsigned long calc_core_pll_rate(struct clk_hw *pll_hw,
- unsigned long target_rate,
- int *pdiv_prim, int *pdiv_clk)
- {
- static const int prim_divs[] = {
- 2, 3, 4, 5, 6, 7, 8, 9, 10, 12, 14, 15, 16,
- 18, 20, 21, 24, 25, 28, 30, 35, 36, 42, 49,
- };
- const unsigned long xosc_rate = clk_hw_get_rate(clk_xosc);
- const unsigned long core_min = xosc_rate * 16;
- const unsigned long core_max = 2400000000;
- int best_div_prim = 1, best_div_clk = 1;
- unsigned long best_rate = core_max + 1;
- unsigned long core_rate = 0;
- int div_int, div_frac;
- u64 div;
- int i;
- /* Given the target rate, choose a set of divisors/multipliers */
- for (i = 0; i < ARRAY_SIZE(prim_divs); i++) {
- int div_prim = prim_divs[i];
- int div_clk;
- for (div_clk = 1; div_clk <= 256; div_clk++) {
- core_rate = target_rate * div_clk * div_prim;
- if (core_rate >= core_min) {
- if (core_rate < best_rate) {
- best_rate = core_rate;
- best_div_prim = div_prim;
- best_div_clk = div_clk;
- }
- break;
- }
- }
- }
- if (best_rate < core_max) {
- div = ((best_rate << 24) + xosc_rate / 2) / xosc_rate;
- div_int = div >> 24;
- div_frac = div % (1 << 24);
- core_rate = (xosc_rate * ((div_int << 24) + div_frac) + (1 << 23)) >> 24;
- } else {
- core_rate = 0;
- }
- if (pdiv_prim)
- *pdiv_prim = best_div_prim;
- if (pdiv_clk)
- *pdiv_clk = best_div_clk;
- return core_rate;
- }
- static void rp1_clock_choose_div_and_prate(struct clk_hw *hw,
- int parent_idx,
- unsigned long rate,
- unsigned long *prate,
- unsigned long *calc_rate)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- const struct rp1_clock_data *data = clock->data;
- struct clk_hw *clk_audio_hw = &clk_audio->hw;
- struct clk_hw *clk_i2s_hw = &clk_i2s->hw;
- struct clk_hw *parent;
- u32 div;
- u64 tmp;
- parent = clk_hw_get_parent_by_index(hw, parent_idx);
- if (hw == clk_i2s_hw && clk_i2s->cached_rate == rate && parent == clk_audio_hw) {
- *prate = clk_audio->cached_rate;
- *calc_rate = rate;
- return;
- }
- if (hw == clk_i2s_hw && parent == clk_audio_hw) {
- unsigned long core_rate, audio_rate, i2s_rate;
- int div_prim, div_clk;
- core_rate = calc_core_pll_rate(parent, rate, &div_prim, &div_clk);
- audio_rate = DIV_ROUND_CLOSEST(core_rate, div_prim);
- i2s_rate = DIV_ROUND_CLOSEST(audio_rate, div_clk);
- clk_audio_core->cached_rate = core_rate;
- clk_audio->cached_rate = audio_rate;
- clk_i2s->cached_rate = i2s_rate;
- *prate = audio_rate;
- *calc_rate = i2s_rate;
- return;
- }
- *prate = clk_hw_get_rate(parent);
- div = rp1_clock_choose_div(rate, *prate, data);
- if (!div) {
- *calc_rate = 0;
- return;
- }
- /* Recalculate to account for rounding errors */
- tmp = (u64)*prate << CLK_DIV_FRAC_BITS;
- tmp = div_u64(tmp, div);
- /*
- * Prevent overclocks - if all parent choices result in
- * a downstream clock in excess of the maximum, then the
- * call to set the clock will fail.
- */
- if (tmp > data->max_freq)
- *calc_rate = 0;
- else
- *calc_rate = tmp;
- }
- static int rp1_clock_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- struct clk_hw *parent, *best_parent = NULL;
- unsigned long best_rate = 0;
- unsigned long best_prate = 0;
- unsigned long best_rate_diff = ULONG_MAX;
- unsigned long prate, calc_rate;
- size_t i;
- /*
- * If the NO_REPARENT flag is set, try to use existing parent.
- */
- if ((clk_hw_get_flags(hw) & CLK_SET_RATE_NO_REPARENT)) {
- i = rp1_clock_get_parent(hw);
- parent = clk_hw_get_parent_by_index(hw, i);
- if (parent) {
- rp1_clock_choose_div_and_prate(hw, i, req->rate, &prate,
- &calc_rate);
- if (calc_rate > 0) {
- req->best_parent_hw = parent;
- req->best_parent_rate = prate;
- req->rate = calc_rate;
- return 0;
- }
- }
- }
- /*
- * Select parent clock that results in the closest rate (lower or
- * higher)
- */
- for (i = 0; i < clk_hw_get_num_parents(hw); i++) {
- parent = clk_hw_get_parent_by_index(hw, i);
- if (!parent)
- continue;
- rp1_clock_choose_div_and_prate(hw, i, req->rate, &prate,
- &calc_rate);
- if (abs_diff(calc_rate, req->rate) < best_rate_diff) {
- best_parent = parent;
- best_prate = prate;
- best_rate = calc_rate;
- best_rate_diff = abs_diff(calc_rate, req->rate);
- if (best_rate_diff == 0)
- break;
- }
- }
- if (best_rate == 0)
- return -EINVAL;
- req->best_parent_hw = best_parent;
- req->best_parent_rate = best_prate;
- req->rate = best_rate;
- return 0;
- }
- static int rp1_varsrc_set_rate(struct clk_hw *hw,
- unsigned long rate, unsigned long parent_rate)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- /*
- * "varsrc" exists purely to let clock dividers know the frequency
- * of an externally-managed clock source (such as MIPI DSI byte-clock)
- * which may change at run-time as a side-effect of some other driver.
- */
- clock->cached_rate = rate;
- return 0;
- }
- static unsigned long rp1_varsrc_recalc_rate(struct clk_hw *hw,
- unsigned long parent_rate)
- {
- struct rp1_clk_desc *clock = container_of(hw, struct rp1_clk_desc, hw);
- return clock->cached_rate;
- }
- static int rp1_varsrc_determine_rate(struct clk_hw *hw,
- struct clk_rate_request *req)
- {
- return 0;
- }
- static const struct clk_ops rp1_pll_core_ops = {
- .is_prepared = rp1_pll_core_is_on,
- .prepare = rp1_pll_core_on,
- .unprepare = rp1_pll_core_off,
- .set_rate = rp1_pll_core_set_rate,
- .recalc_rate = rp1_pll_core_recalc_rate,
- .determine_rate = rp1_pll_core_determine_rate,
- };
- static const struct clk_ops rp1_pll_ops = {
- .set_rate = rp1_pll_set_rate,
- .recalc_rate = rp1_pll_recalc_rate,
- .determine_rate = rp1_pll_determine_rate,
- };
- static const struct clk_ops rp1_pll_ph_ops = {
- .is_prepared = rp1_pll_ph_is_on,
- .prepare = rp1_pll_ph_on,
- .unprepare = rp1_pll_ph_off,
- .recalc_rate = rp1_pll_ph_recalc_rate,
- .determine_rate = rp1_pll_ph_determine_rate,
- };
- static const struct clk_ops rp1_pll_divider_ops = {
- .is_prepared = rp1_pll_divider_is_on,
- .prepare = rp1_pll_divider_on,
- .unprepare = rp1_pll_divider_off,
- .set_rate = rp1_pll_divider_set_rate,
- .recalc_rate = rp1_pll_divider_recalc_rate,
- .determine_rate = rp1_pll_divider_determine_rate,
- };
- static const struct clk_ops rp1_clk_ops = {
- .is_prepared = rp1_clock_is_on,
- .prepare = rp1_clock_on,
- .unprepare = rp1_clock_off,
- .recalc_rate = rp1_clock_recalc_rate,
- .get_parent = rp1_clock_get_parent,
- .set_parent = rp1_clock_set_parent,
- .set_rate_and_parent = rp1_clock_set_rate_and_parent,
- .set_rate = rp1_clock_set_rate,
- .determine_rate = rp1_clock_determine_rate,
- };
- static const struct clk_ops rp1_varsrc_ops = {
- .set_rate = rp1_varsrc_set_rate,
- .recalc_rate = rp1_varsrc_recalc_rate,
- .determine_rate = rp1_varsrc_determine_rate,
- };
- static struct clk_hw *rp1_register_pll(struct rp1_clockman *clockman,
- struct rp1_clk_desc *desc)
- {
- int ret;
- desc->clockman = clockman;
- ret = devm_clk_hw_register(clockman->dev, &desc->hw);
- if (ret)
- return ERR_PTR(ret);
- return &desc->hw;
- }
- static struct clk_hw *rp1_register_pll_divider(struct rp1_clockman *clockman,
- struct rp1_clk_desc *desc)
- {
- const struct rp1_pll_data *divider_data = desc->data;
- int ret;
- desc->div.reg = clockman->regs + divider_data->ctrl_reg;
- desc->div.shift = __ffs(PLL_SEC_DIV_MASK);
- desc->div.width = __ffs(~(PLL_SEC_DIV_MASK >> desc->div.shift));
- desc->div.flags = CLK_DIVIDER_ROUND_CLOSEST;
- desc->div.lock = &clockman->regs_lock;
- desc->div.hw.init = desc->hw.init;
- desc->div.table = pll_sec_div_table;
- desc->clockman = clockman;
- ret = devm_clk_hw_register(clockman->dev, &desc->div.hw);
- if (ret)
- return ERR_PTR(ret);
- return &desc->div.hw;
- }
- static struct clk_hw *rp1_register_clock(struct rp1_clockman *clockman,
- struct rp1_clk_desc *desc)
- {
- const struct rp1_clock_data *clock_data = desc->data;
- int ret;
- if (WARN_ON_ONCE(MAX_CLK_PARENTS <
- clock_data->num_std_parents + clock_data->num_aux_parents))
- return ERR_PTR(-EINVAL);
- /* There must be a gap for the AUX selector */
- if (WARN_ON_ONCE(clock_data->num_std_parents > AUX_SEL &&
- desc->hw.init->parent_data[AUX_SEL].index != -1))
- return ERR_PTR(-EINVAL);
- desc->clockman = clockman;
- ret = devm_clk_hw_register(clockman->dev, &desc->hw);
- if (ret)
- return ERR_PTR(ret);
- return &desc->hw;
- }
- /* Assignment helper macros for different clock types. */
- #define _REGISTER(f, ...) { .clk_register = f, __VA_ARGS__ }
- #define CLK_DATA(type, ...) .data = &(struct type) { __VA_ARGS__ }
- #define REGISTER_PLL(...) _REGISTER(&rp1_register_pll, \
- __VA_ARGS__)
- #define REGISTER_PLL_DIV(...) _REGISTER(&rp1_register_pll_divider, \
- __VA_ARGS__)
- #define REGISTER_CLK(...) _REGISTER(&rp1_register_clock, \
- __VA_ARGS__)
- static struct rp1_clk_desc pll_sys_core_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_sys_core",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_pll_core_ops,
- CLK_IS_CRITICAL
- ),
- CLK_DATA(rp1_pll_core_data,
- .cs_reg = PLL_SYS_CS,
- .pwr_reg = PLL_SYS_PWR,
- .fbdiv_int_reg = PLL_SYS_FBDIV_INT,
- .fbdiv_frac_reg = PLL_SYS_FBDIV_FRAC,
- )
- );
- static struct rp1_clk_desc pll_audio_core_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_audio_core",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_pll_core_ops,
- CLK_IS_CRITICAL
- ),
- CLK_DATA(rp1_pll_core_data,
- .cs_reg = PLL_AUDIO_CS,
- .pwr_reg = PLL_AUDIO_PWR,
- .fbdiv_int_reg = PLL_AUDIO_FBDIV_INT,
- .fbdiv_frac_reg = PLL_AUDIO_FBDIV_FRAC,
- )
- );
- static struct rp1_clk_desc pll_video_core_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_video_core",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_pll_core_ops,
- CLK_IS_CRITICAL
- ),
- CLK_DATA(rp1_pll_core_data,
- .cs_reg = PLL_VIDEO_CS,
- .pwr_reg = PLL_VIDEO_PWR,
- .fbdiv_int_reg = PLL_VIDEO_FBDIV_INT,
- .fbdiv_frac_reg = PLL_VIDEO_FBDIV_FRAC,
- )
- );
- static struct rp1_clk_desc pll_sys_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_sys",
- (const struct clk_parent_data[]) {
- { .hw = &pll_sys_core_desc.hw }
- },
- &rp1_pll_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_SYS_PRIM,
- .fc0_src = FC_NUM(0, 2),
- )
- );
- static struct rp1_clk_desc pll_audio_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_audio",
- (const struct clk_parent_data[]) {
- { .hw = &pll_audio_core_desc.hw }
- },
- &rp1_pll_ops,
- CLK_SET_RATE_PARENT
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_AUDIO_PRIM,
- .fc0_src = FC_NUM(4, 2),
- )
- );
- static struct rp1_clk_desc pll_video_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_video",
- (const struct clk_parent_data[]) {
- { .hw = &pll_video_core_desc.hw }
- },
- &rp1_pll_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_VIDEO_PRIM,
- .fc0_src = FC_NUM(3, 2),
- )
- );
- static struct rp1_clk_desc pll_sys_sec_desc = REGISTER_PLL_DIV(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_sys_sec",
- (const struct clk_parent_data[]) {
- { .hw = &pll_sys_core_desc.hw }
- },
- &rp1_pll_divider_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_SYS_SEC,
- .fc0_src = FC_NUM(2, 2),
- )
- );
- static struct rp1_clk_desc pll_video_sec_desc = REGISTER_PLL_DIV(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_video_sec",
- (const struct clk_parent_data[]) {
- { .hw = &pll_video_core_desc.hw }
- },
- &rp1_pll_divider_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_VIDEO_SEC,
- .fc0_src = FC_NUM(5, 3),
- )
- );
- static const struct clk_parent_data clk_eth_tsu_parents[] = {
- { .index = 0 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_eth_tsu_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_eth_tsu",
- clk_eth_tsu_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 8,
- .ctrl_reg = CLK_ETH_TSU_CTRL,
- .div_int_reg = CLK_ETH_TSU_DIV_INT,
- .sel_reg = CLK_ETH_TSU_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(5, 7),
- )
- );
- static const struct clk_parent_data clk_eth_parents[] = {
- { .hw = &pll_sys_sec_desc.div.hw },
- { .hw = &pll_sys_desc.hw },
- { .hw = &pll_video_sec_desc.hw },
- };
- static struct rp1_clk_desc clk_eth_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_eth",
- clk_eth_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_ETH_CTRL,
- .div_int_reg = CLK_ETH_DIV_INT,
- .sel_reg = CLK_ETH_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 125 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(4, 6),
- )
- );
- static const struct clk_parent_data clk_sys_parents[] = {
- { .index = 0 },
- { .index = -1 },
- { .hw = &pll_sys_desc.hw },
- };
- static struct rp1_clk_desc clk_sys_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_sys",
- clk_sys_parents,
- &rp1_clk_ops,
- CLK_IS_CRITICAL
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 3,
- .num_aux_parents = 0,
- .ctrl_reg = CLK_SYS_CTRL,
- .div_int_reg = CLK_SYS_DIV_INT,
- .sel_reg = CLK_SYS_SEL,
- .div_int_max = DIV_INT_24BIT_MAX,
- .max_freq = 200 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(0, 4),
- .clk_src_mask = 0x3,
- )
- );
- static struct rp1_clk_desc pll_sys_pri_ph_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_sys_pri_ph",
- (const struct clk_parent_data[]) {
- { .hw = &pll_sys_desc.hw }
- },
- &rp1_pll_ph_ops,
- 0
- ),
- CLK_DATA(rp1_pll_ph_data,
- .ph_reg = PLL_SYS_PRIM,
- .fixed_divider = 2,
- .phase = RP1_PLL_PHASE_0,
- .fc0_src = FC_NUM(1, 2),
- )
- );
- static struct rp1_clk_desc pll_audio_pri_ph_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_audio_pri_ph",
- (const struct clk_parent_data[]) {
- { .hw = &pll_audio_desc.hw }
- },
- &rp1_pll_ph_ops,
- 0
- ),
- CLK_DATA(rp1_pll_ph_data,
- .ph_reg = PLL_AUDIO_PRIM,
- .fixed_divider = 2,
- .phase = RP1_PLL_PHASE_0,
- .fc0_src = FC_NUM(5, 1),
- )
- );
- static struct rp1_clk_desc pll_video_pri_ph_desc = REGISTER_PLL(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_video_pri_ph",
- (const struct clk_parent_data[]) {
- { .hw = &pll_video_desc.hw }
- },
- &rp1_pll_ph_ops,
- 0
- ),
- CLK_DATA(rp1_pll_ph_data,
- .ph_reg = PLL_VIDEO_PRIM,
- .fixed_divider = 2,
- .phase = RP1_PLL_PHASE_0,
- .fc0_src = FC_NUM(4, 3),
- )
- );
- static struct rp1_clk_desc pll_audio_sec_desc = REGISTER_PLL_DIV(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_audio_sec",
- (const struct clk_parent_data[]) {
- { .hw = &pll_audio_core_desc.hw }
- },
- &rp1_pll_divider_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_AUDIO_SEC,
- .fc0_src = FC_NUM(6, 2),
- )
- );
- static struct rp1_clk_desc pll_audio_tern_desc = REGISTER_PLL_DIV(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "pll_audio_tern",
- (const struct clk_parent_data[]) {
- { .hw = &pll_audio_core_desc.hw }
- },
- &rp1_pll_divider_ops,
- 0
- ),
- CLK_DATA(rp1_pll_data,
- .ctrl_reg = PLL_AUDIO_TERN,
- .fc0_src = FC_NUM(6, 2),
- )
- );
- static struct rp1_clk_desc clk_slow_sys_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_slow_sys",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_clk_ops,
- CLK_IS_CRITICAL
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 1,
- .num_aux_parents = 0,
- .ctrl_reg = CLK_SLOW_SYS_CTRL,
- .div_int_reg = CLK_SLOW_SYS_DIV_INT,
- .sel_reg = CLK_SLOW_SYS_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(1, 4),
- .clk_src_mask = 0x1,
- )
- );
- static const struct clk_parent_data clk_dma_parents[] = {
- { .hw = &pll_sys_pri_ph_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_dma_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_dma",
- clk_dma_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_DMA_CTRL,
- .div_int_reg = CLK_DMA_DIV_INT,
- .sel_reg = CLK_DMA_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(2, 2),
- )
- );
- static const struct clk_parent_data clk_uart_parents[] = {
- { .hw = &pll_sys_pri_ph_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_uart_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_uart",
- clk_uart_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_UART_CTRL,
- .div_int_reg = CLK_UART_DIV_INT,
- .sel_reg = CLK_UART_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(6, 7),
- )
- );
- static const struct clk_parent_data clk_pwm0_parents[] = {
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_pwm0_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_pwm0",
- clk_pwm0_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_PWM0_CTRL,
- .div_int_reg = CLK_PWM0_DIV_INT,
- .div_frac_reg = CLK_PWM0_DIV_FRAC,
- .sel_reg = CLK_PWM0_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 76800 * HZ_PER_KHZ,
- .fc0_src = FC_NUM(0, 5),
- )
- );
- static const struct clk_parent_data clk_pwm1_parents[] = {
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_pwm1_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_pwm1",
- clk_pwm1_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_PWM1_CTRL,
- .div_int_reg = CLK_PWM1_DIV_INT,
- .div_frac_reg = CLK_PWM1_DIV_FRAC,
- .sel_reg = CLK_PWM1_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 76800 * HZ_PER_KHZ,
- .fc0_src = FC_NUM(1, 5),
- )
- );
- static const struct clk_parent_data clk_audio_in_parents[] = {
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_audio_in_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_audio_in",
- clk_audio_in_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 5,
- .ctrl_reg = CLK_AUDIO_IN_CTRL,
- .div_int_reg = CLK_AUDIO_IN_DIV_INT,
- .sel_reg = CLK_AUDIO_IN_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 76800 * HZ_PER_KHZ,
- .fc0_src = FC_NUM(2, 5),
- )
- );
- static const struct clk_parent_data clk_audio_out_parents[] = {
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = 0 },
- };
- static struct rp1_clk_desc clk_audio_out_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_audio_out",
- clk_audio_out_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 4,
- .ctrl_reg = CLK_AUDIO_OUT_CTRL,
- .div_int_reg = CLK_AUDIO_OUT_DIV_INT,
- .sel_reg = CLK_AUDIO_OUT_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 153600 * HZ_PER_KHZ,
- .fc0_src = FC_NUM(3, 5),
- )
- );
- static const struct clk_parent_data clk_i2s_parents[] = {
- { .index = 0 },
- { .hw = &pll_audio_desc.hw },
- { .hw = &pll_audio_sec_desc.hw },
- };
- static struct rp1_clk_desc clk_i2s_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_i2s",
- clk_i2s_parents,
- &rp1_clk_ops,
- CLK_SET_RATE_PARENT
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 3,
- .ctrl_reg = CLK_I2S_CTRL,
- .div_int_reg = CLK_I2S_DIV_INT,
- .sel_reg = CLK_I2S_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(4, 4),
- )
- );
- static struct rp1_clk_desc clk_mipi0_cfg_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_mipi0_cfg",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 1,
- .ctrl_reg = CLK_MIPI0_CFG_CTRL,
- .div_int_reg = CLK_MIPI0_CFG_DIV_INT,
- .sel_reg = CLK_MIPI0_CFG_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(4, 5),
- )
- );
- static struct rp1_clk_desc clk_mipi1_cfg_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_mipi1_cfg",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 1,
- .ctrl_reg = CLK_MIPI1_CFG_CTRL,
- .div_int_reg = CLK_MIPI1_CFG_DIV_INT,
- .sel_reg = CLK_MIPI1_CFG_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(5, 6),
- .clk_src_mask = 0x1,
- )
- );
- static struct rp1_clk_desc clk_adc_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_adc",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 1,
- .ctrl_reg = CLK_ADC_CTRL,
- .div_int_reg = CLK_ADC_DIV_INT,
- .sel_reg = CLK_ADC_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(5, 5),
- )
- );
- static struct rp1_clk_desc clk_sdio_timer_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_sdio_timer",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 1,
- .ctrl_reg = CLK_SDIO_TIMER_CTRL,
- .div_int_reg = CLK_SDIO_TIMER_DIV_INT,
- .sel_reg = CLK_SDIO_TIMER_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 50 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(3, 4),
- )
- );
- static struct rp1_clk_desc clk_sdio_alt_src_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_sdio_alt_src",
- (const struct clk_parent_data[]) {
- { .hw = &pll_sys_desc.hw }
- },
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 1,
- .ctrl_reg = CLK_SDIO_ALT_SRC_CTRL,
- .div_int_reg = CLK_SDIO_ALT_SRC_DIV_INT,
- .sel_reg = CLK_SDIO_ALT_SRC_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 200 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(5, 4),
- )
- );
- static const struct clk_parent_data clk_dpi_parents[] = {
- { .hw = &pll_sys_desc.hw },
- { .hw = &pll_video_sec_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_dpi_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_dpi",
- clk_dpi_parents,
- &rp1_clk_ops,
- CLK_SET_RATE_NO_REPARENT /* Let DPI driver set parent */
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 8,
- .ctrl_reg = VIDEO_CLK_DPI_CTRL,
- .div_int_reg = VIDEO_CLK_DPI_DIV_INT,
- .sel_reg = VIDEO_CLK_DPI_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 200 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(1, 6),
- )
- );
- static const struct clk_parent_data clk_gp0_parents[] = {
- { .index = 0 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_sys_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_i2s_desc.hw },
- { .hw = &clk_adc_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_sys_desc.hw },
- };
- static struct rp1_clk_desc clk_gp0_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp0",
- clk_gp0_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(0),
- .ctrl_reg = CLK_GP0_CTRL,
- .div_int_reg = CLK_GP0_DIV_INT,
- .div_frac_reg = CLK_GP0_DIV_FRAC,
- .sel_reg = CLK_GP0_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(0, 1),
- )
- );
- static const struct clk_parent_data clk_gp1_parents[] = {
- { .hw = &clk_sdio_timer_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_sys_pri_ph_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_adc_desc.hw },
- { .hw = &clk_dpi_desc.hw },
- { .hw = &clk_pwm0_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_gp1_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp1",
- clk_gp1_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(1),
- .ctrl_reg = CLK_GP1_CTRL,
- .div_int_reg = CLK_GP1_DIV_INT,
- .div_frac_reg = CLK_GP1_DIV_FRAC,
- .sel_reg = CLK_GP1_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(1, 1),
- )
- );
- static struct rp1_clk_desc clksrc_mipi0_dsi_byteclk_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clksrc_mipi0_dsi_byteclk",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_varsrc_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 1,
- .num_aux_parents = 0,
- )
- );
- static struct rp1_clk_desc clksrc_mipi1_dsi_byteclk_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clksrc_mipi1_dsi_byteclk",
- (const struct clk_parent_data[]) { { .index = 0 } },
- &rp1_varsrc_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 1,
- .num_aux_parents = 0,
- )
- );
- static const struct clk_parent_data clk_mipi0_dpi_parents[] = {
- { .hw = &pll_sys_desc.hw },
- { .hw = &pll_video_sec_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .hw = &clksrc_mipi0_dsi_byteclk_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_mipi0_dpi_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_mipi0_dpi",
- clk_mipi0_dpi_parents,
- &rp1_clk_ops,
- CLK_SET_RATE_NO_REPARENT /* Let DSI driver set parent */
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 8,
- .ctrl_reg = VIDEO_CLK_MIPI0_DPI_CTRL,
- .div_int_reg = VIDEO_CLK_MIPI0_DPI_DIV_INT,
- .div_frac_reg = VIDEO_CLK_MIPI0_DPI_DIV_FRAC,
- .sel_reg = VIDEO_CLK_MIPI0_DPI_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 200 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(2, 6),
- )
- );
- static const struct clk_parent_data clk_mipi1_dpi_parents[] = {
- { .hw = &pll_sys_desc.hw },
- { .hw = &pll_video_sec_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .hw = &clksrc_mipi1_dsi_byteclk_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_mipi1_dpi_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_mipi1_dpi",
- clk_mipi1_dpi_parents,
- &rp1_clk_ops,
- CLK_SET_RATE_NO_REPARENT /* Let DSI driver set parent */
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 8,
- .ctrl_reg = VIDEO_CLK_MIPI1_DPI_CTRL,
- .div_int_reg = VIDEO_CLK_MIPI1_DPI_DIV_INT,
- .div_frac_reg = VIDEO_CLK_MIPI1_DPI_DIV_FRAC,
- .sel_reg = VIDEO_CLK_MIPI1_DPI_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 200 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(3, 6),
- )
- );
- static const struct clk_parent_data clk_gp2_parents[] = {
- { .hw = &clk_sdio_alt_src_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_sys_sec_desc.hw },
- { .index = -1 },
- { .hw = &pll_video_desc.hw },
- { .hw = &clk_audio_in_desc.hw },
- { .hw = &clk_dpi_desc.hw },
- { .hw = &clk_pwm0_desc.hw },
- { .hw = &clk_pwm1_desc.hw },
- { .hw = &clk_mipi0_dpi_desc.hw },
- { .hw = &clk_mipi1_cfg_desc.hw },
- { .hw = &clk_sys_desc.hw },
- };
- static struct rp1_clk_desc clk_gp2_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp2",
- clk_gp2_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(2),
- .ctrl_reg = CLK_GP2_CTRL,
- .div_int_reg = CLK_GP2_DIV_INT,
- .div_frac_reg = CLK_GP2_DIV_FRAC,
- .sel_reg = CLK_GP2_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(2, 1),
- )
- );
- static const struct clk_parent_data clk_gp3_parents[] = {
- { .index = 0 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_video_pri_ph_desc.hw },
- { .hw = &clk_audio_out_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_mipi1_dpi_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_gp3_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp3",
- clk_gp3_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(3),
- .ctrl_reg = CLK_GP3_CTRL,
- .div_int_reg = CLK_GP3_DIV_INT,
- .div_frac_reg = CLK_GP3_DIV_FRAC,
- .sel_reg = CLK_GP3_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(3, 1),
- )
- );
- static const struct clk_parent_data clk_gp4_parents[] = {
- { .index = 0 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_mipi0_cfg_desc.hw },
- { .hw = &clk_uart_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .hw = &clk_sys_desc.hw },
- };
- static struct rp1_clk_desc clk_gp4_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp4",
- clk_gp4_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(4),
- .ctrl_reg = CLK_GP4_CTRL,
- .div_int_reg = CLK_GP4_DIV_INT,
- .div_frac_reg = CLK_GP4_DIV_FRAC,
- .sel_reg = CLK_GP4_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(4, 1),
- )
- );
- static const struct clk_parent_data clk_vec_parents[] = {
- { .hw = &pll_sys_pri_ph_desc.hw },
- { .hw = &pll_video_sec_desc.hw },
- { .hw = &pll_video_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_vec_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_vec",
- clk_vec_parents,
- &rp1_clk_ops,
- CLK_SET_RATE_NO_REPARENT /* Let VEC driver set parent */
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 8,
- .ctrl_reg = VIDEO_CLK_VEC_CTRL,
- .div_int_reg = VIDEO_CLK_VEC_DIV_INT,
- .sel_reg = VIDEO_CLK_VEC_SEL,
- .div_int_max = DIV_INT_8BIT_MAX,
- .max_freq = 108 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(0, 6),
- )
- );
- static const struct clk_parent_data clk_gp5_parents[] = {
- { .index = 0 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .hw = &pll_video_sec_desc.hw },
- { .hw = &clk_eth_tsu_desc.hw },
- { .index = -1 },
- { .hw = &clk_vec_desc.hw },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- { .index = -1 },
- };
- static struct rp1_clk_desc clk_gp5_desc = REGISTER_CLK(
- .hw.init = CLK_HW_INIT_PARENTS_DATA(
- "clk_gp5",
- clk_gp5_parents,
- &rp1_clk_ops,
- 0
- ),
- CLK_DATA(rp1_clock_data,
- .num_std_parents = 0,
- .num_aux_parents = 16,
- .oe_mask = BIT(5),
- .ctrl_reg = CLK_GP5_CTRL,
- .div_int_reg = CLK_GP5_DIV_INT,
- .div_frac_reg = CLK_GP5_DIV_FRAC,
- .sel_reg = CLK_GP5_SEL,
- .div_int_max = DIV_INT_16BIT_MAX,
- .max_freq = 100 * HZ_PER_MHZ,
- .fc0_src = FC_NUM(5, 1),
- )
- );
- static struct rp1_clk_desc *const clk_desc_array[] = {
- [RP1_PLL_SYS_CORE] = &pll_sys_core_desc,
- [RP1_PLL_AUDIO_CORE] = &pll_audio_core_desc,
- [RP1_PLL_VIDEO_CORE] = &pll_video_core_desc,
- [RP1_PLL_SYS] = &pll_sys_desc,
- [RP1_CLK_ETH_TSU] = &clk_eth_tsu_desc,
- [RP1_CLK_ETH] = &clk_eth_desc,
- [RP1_CLK_SYS] = &clk_sys_desc,
- [RP1_PLL_SYS_PRI_PH] = &pll_sys_pri_ph_desc,
- [RP1_PLL_SYS_SEC] = &pll_sys_sec_desc,
- [RP1_PLL_AUDIO] = &pll_audio_desc,
- [RP1_PLL_VIDEO] = &pll_video_desc,
- [RP1_PLL_AUDIO_PRI_PH] = &pll_audio_pri_ph_desc,
- [RP1_PLL_VIDEO_PRI_PH] = &pll_video_pri_ph_desc,
- [RP1_PLL_AUDIO_SEC] = &pll_audio_sec_desc,
- [RP1_PLL_VIDEO_SEC] = &pll_video_sec_desc,
- [RP1_PLL_AUDIO_TERN] = &pll_audio_tern_desc,
- [RP1_CLK_SLOW_SYS] = &clk_slow_sys_desc,
- [RP1_CLK_DMA] = &clk_dma_desc,
- [RP1_CLK_UART] = &clk_uart_desc,
- [RP1_CLK_PWM0] = &clk_pwm0_desc,
- [RP1_CLK_PWM1] = &clk_pwm1_desc,
- [RP1_CLK_AUDIO_IN] = &clk_audio_in_desc,
- [RP1_CLK_AUDIO_OUT] = &clk_audio_out_desc,
- [RP1_CLK_I2S] = &clk_i2s_desc,
- [RP1_CLK_MIPI0_CFG] = &clk_mipi0_cfg_desc,
- [RP1_CLK_MIPI1_CFG] = &clk_mipi1_cfg_desc,
- [RP1_CLK_ADC] = &clk_adc_desc,
- [RP1_CLK_SDIO_TIMER] = &clk_sdio_timer_desc,
- [RP1_CLK_SDIO_ALT_SRC] = &clk_sdio_alt_src_desc,
- [RP1_CLK_GP0] = &clk_gp0_desc,
- [RP1_CLK_GP1] = &clk_gp1_desc,
- [RP1_CLK_GP2] = &clk_gp2_desc,
- [RP1_CLK_GP3] = &clk_gp3_desc,
- [RP1_CLK_GP4] = &clk_gp4_desc,
- [RP1_CLK_GP5] = &clk_gp5_desc,
- [RP1_CLK_VEC] = &clk_vec_desc,
- [RP1_CLK_DPI] = &clk_dpi_desc,
- [RP1_CLK_MIPI0_DPI] = &clk_mipi0_dpi_desc,
- [RP1_CLK_MIPI1_DPI] = &clk_mipi1_dpi_desc,
- [RP1_CLK_MIPI0_DSI_BYTECLOCK] = &clksrc_mipi0_dsi_byteclk_desc,
- [RP1_CLK_MIPI1_DSI_BYTECLOCK] = &clksrc_mipi1_dsi_byteclk_desc,
- };
- static const struct regmap_range rp1_reg_ranges[] = {
- regmap_reg_range(PLL_SYS_CS, PLL_SYS_SEC),
- regmap_reg_range(PLL_AUDIO_CS, PLL_AUDIO_TERN),
- regmap_reg_range(PLL_VIDEO_CS, PLL_VIDEO_SEC),
- regmap_reg_range(GPCLK_OE_CTRL, GPCLK_OE_CTRL),
- regmap_reg_range(CLK_SYS_CTRL, CLK_SYS_DIV_INT),
- regmap_reg_range(CLK_SYS_SEL, CLK_SYS_SEL),
- regmap_reg_range(CLK_SLOW_SYS_CTRL, CLK_SLOW_SYS_DIV_INT),
- regmap_reg_range(CLK_SLOW_SYS_SEL, CLK_SLOW_SYS_SEL),
- regmap_reg_range(CLK_DMA_CTRL, CLK_DMA_DIV_INT),
- regmap_reg_range(CLK_DMA_SEL, CLK_DMA_SEL),
- regmap_reg_range(CLK_UART_CTRL, CLK_UART_DIV_INT),
- regmap_reg_range(CLK_UART_SEL, CLK_UART_SEL),
- regmap_reg_range(CLK_ETH_CTRL, CLK_ETH_DIV_INT),
- regmap_reg_range(CLK_ETH_SEL, CLK_ETH_SEL),
- regmap_reg_range(CLK_PWM0_CTRL, CLK_PWM0_SEL),
- regmap_reg_range(CLK_PWM1_CTRL, CLK_PWM1_SEL),
- regmap_reg_range(CLK_AUDIO_IN_CTRL, CLK_AUDIO_IN_DIV_INT),
- regmap_reg_range(CLK_AUDIO_IN_SEL, CLK_AUDIO_IN_SEL),
- regmap_reg_range(CLK_AUDIO_OUT_CTRL, CLK_AUDIO_OUT_DIV_INT),
- regmap_reg_range(CLK_AUDIO_OUT_SEL, CLK_AUDIO_OUT_SEL),
- regmap_reg_range(CLK_I2S_CTRL, CLK_I2S_DIV_INT),
- regmap_reg_range(CLK_I2S_SEL, CLK_I2S_SEL),
- regmap_reg_range(CLK_MIPI0_CFG_CTRL, CLK_MIPI0_CFG_DIV_INT),
- regmap_reg_range(CLK_MIPI0_CFG_SEL, CLK_MIPI0_CFG_SEL),
- regmap_reg_range(CLK_MIPI1_CFG_CTRL, CLK_MIPI1_CFG_DIV_INT),
- regmap_reg_range(CLK_MIPI1_CFG_SEL, CLK_MIPI1_CFG_SEL),
- regmap_reg_range(CLK_PCIE_AUX_CTRL, CLK_PCIE_AUX_DIV_INT),
- regmap_reg_range(CLK_PCIE_AUX_SEL, CLK_PCIE_AUX_SEL),
- regmap_reg_range(CLK_USBH0_MICROFRAME_CTRL, CLK_USBH0_MICROFRAME_DIV_INT),
- regmap_reg_range(CLK_USBH0_MICROFRAME_SEL, CLK_USBH0_MICROFRAME_SEL),
- regmap_reg_range(CLK_USBH1_MICROFRAME_CTRL, CLK_USBH1_MICROFRAME_DIV_INT),
- regmap_reg_range(CLK_USBH1_MICROFRAME_SEL, CLK_USBH1_MICROFRAME_SEL),
- regmap_reg_range(CLK_USBH0_SUSPEND_CTRL, CLK_USBH0_SUSPEND_DIV_INT),
- regmap_reg_range(CLK_USBH0_SUSPEND_SEL, CLK_USBH0_SUSPEND_SEL),
- regmap_reg_range(CLK_USBH1_SUSPEND_CTRL, CLK_USBH1_SUSPEND_DIV_INT),
- regmap_reg_range(CLK_USBH1_SUSPEND_SEL, CLK_USBH1_SUSPEND_SEL),
- regmap_reg_range(CLK_ETH_TSU_CTRL, CLK_ETH_TSU_DIV_INT),
- regmap_reg_range(CLK_ETH_TSU_SEL, CLK_ETH_TSU_SEL),
- regmap_reg_range(CLK_ADC_CTRL, CLK_ADC_DIV_INT),
- regmap_reg_range(CLK_ADC_SEL, CLK_ADC_SEL),
- regmap_reg_range(CLK_SDIO_TIMER_CTRL, CLK_SDIO_TIMER_DIV_INT),
- regmap_reg_range(CLK_SDIO_TIMER_SEL, CLK_SDIO_TIMER_SEL),
- regmap_reg_range(CLK_SDIO_ALT_SRC_CTRL, CLK_SDIO_ALT_SRC_DIV_INT),
- regmap_reg_range(CLK_SDIO_ALT_SRC_SEL, CLK_SDIO_ALT_SRC_SEL),
- regmap_reg_range(CLK_GP0_CTRL, CLK_GP0_SEL),
- regmap_reg_range(CLK_GP1_CTRL, CLK_GP1_SEL),
- regmap_reg_range(CLK_GP2_CTRL, CLK_GP2_SEL),
- regmap_reg_range(CLK_GP3_CTRL, CLK_GP3_SEL),
- regmap_reg_range(CLK_GP4_CTRL, CLK_GP4_SEL),
- regmap_reg_range(CLK_GP5_CTRL, CLK_GP5_SEL),
- regmap_reg_range(CLK_SYS_RESUS_CTRL, CLK_SYS_RESUS_CTRL),
- regmap_reg_range(CLK_SLOW_SYS_RESUS_CTRL, CLK_SLOW_SYS_RESUS_CTRL),
- regmap_reg_range(FC0_REF_KHZ, FC0_RESULT),
- regmap_reg_range(VIDEO_CLK_VEC_CTRL, VIDEO_CLK_VEC_DIV_INT),
- regmap_reg_range(VIDEO_CLK_VEC_SEL, VIDEO_CLK_DPI_DIV_INT),
- regmap_reg_range(VIDEO_CLK_DPI_SEL, VIDEO_CLK_MIPI1_DPI_SEL),
- };
- static const struct regmap_access_table rp1_reg_table = {
- .yes_ranges = rp1_reg_ranges,
- .n_yes_ranges = ARRAY_SIZE(rp1_reg_ranges),
- };
- static const struct regmap_config rp1_clk_regmap_cfg = {
- .reg_bits = 32,
- .val_bits = 32,
- .reg_stride = 4,
- .max_register = PLL_VIDEO_SEC,
- .name = "rp1-clk",
- .rd_table = &rp1_reg_table,
- .disable_locking = true,
- };
- static int rp1_clk_probe(struct platform_device *pdev)
- {
- const size_t asize = ARRAY_SIZE(clk_desc_array);
- struct rp1_clk_desc *desc;
- struct device *dev = &pdev->dev;
- struct rp1_clockman *clockman;
- struct clk_hw **hws;
- unsigned int i;
- clockman = devm_kzalloc(dev, struct_size(clockman, onecell.hws, asize),
- GFP_KERNEL);
- if (!clockman)
- return -ENOMEM;
- spin_lock_init(&clockman->regs_lock);
- clockman->dev = dev;
- clockman->regs = devm_platform_ioremap_resource(pdev, 0);
- if (IS_ERR(clockman->regs))
- return PTR_ERR(clockman->regs);
- clockman->regmap = devm_regmap_init_mmio(dev, clockman->regs,
- &rp1_clk_regmap_cfg);
- if (IS_ERR(clockman->regmap)) {
- dev_err_probe(dev, PTR_ERR(clockman->regmap),
- "could not init clock regmap\n");
- return PTR_ERR(clockman->regmap);
- }
- clockman->onecell.num = asize;
- hws = clockman->onecell.hws;
- for (i = 0; i < asize; i++) {
- desc = clk_desc_array[i];
- if (desc && desc->clk_register && desc->data)
- hws[i] = desc->clk_register(clockman, desc);
- }
- clk_audio_core = &pll_audio_core_desc;
- clk_audio = &pll_audio_desc;
- clk_i2s = &clk_i2s_desc;
- clk_xosc = clk_hw_get_parent_by_index(&clk_i2s->hw, 0);
- platform_set_drvdata(pdev, clockman);
- return devm_of_clk_add_hw_provider(dev, of_clk_hw_onecell_get,
- &clockman->onecell);
- }
- static const struct of_device_id rp1_clk_of_match[] = {
- { .compatible = "raspberrypi,rp1-clocks" },
- {}
- };
- MODULE_DEVICE_TABLE(of, rp1_clk_of_match);
- static struct platform_driver rp1_clk_driver = {
- .driver = {
- .name = "rp1-clk",
- .of_match_table = rp1_clk_of_match,
- },
- .probe = rp1_clk_probe,
- };
- module_platform_driver(rp1_clk_driver);
- MODULE_AUTHOR("Naushir Patuck <naush@raspberrypi.com>");
- MODULE_AUTHOR("Andrea della Porta <andrea.porta@suse.com>");
- MODULE_DESCRIPTION("RP1 clock driver");
- MODULE_LICENSE("GPL");
|