162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (c) 2011 Samsung Electronics Co., Ltd. 462306a36Sopenharmony_ci * http://www.samsung.com/ 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * samsung - Common hr-timer support (s3c and s5p) 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#include <linux/interrupt.h> 1062306a36Sopenharmony_ci#include <linux/irq.h> 1162306a36Sopenharmony_ci#include <linux/err.h> 1262306a36Sopenharmony_ci#include <linux/clk.h> 1362306a36Sopenharmony_ci#include <linux/clockchips.h> 1462306a36Sopenharmony_ci#include <linux/list.h> 1562306a36Sopenharmony_ci#include <linux/module.h> 1662306a36Sopenharmony_ci#include <linux/of.h> 1762306a36Sopenharmony_ci#include <linux/of_address.h> 1862306a36Sopenharmony_ci#include <linux/of_irq.h> 1962306a36Sopenharmony_ci#include <linux/platform_device.h> 2062306a36Sopenharmony_ci#include <linux/slab.h> 2162306a36Sopenharmony_ci#include <linux/sched_clock.h> 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci#include <clocksource/samsung_pwm.h> 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci/* 2662306a36Sopenharmony_ci * Clocksource driver 2762306a36Sopenharmony_ci */ 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci#define REG_TCFG0 0x00 3062306a36Sopenharmony_ci#define REG_TCFG1 0x04 3162306a36Sopenharmony_ci#define REG_TCON 0x08 3262306a36Sopenharmony_ci#define REG_TINT_CSTAT 0x44 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci#define REG_TCNTB(chan) (0x0c + 12 * (chan)) 3562306a36Sopenharmony_ci#define REG_TCMPB(chan) (0x10 + 12 * (chan)) 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci#define TCFG0_PRESCALER_MASK 0xff 3862306a36Sopenharmony_ci#define TCFG0_PRESCALER1_SHIFT 8 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci#define TCFG1_SHIFT(x) ((x) * 4) 4162306a36Sopenharmony_ci#define TCFG1_MUX_MASK 0xf 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci/* 4462306a36Sopenharmony_ci * Each channel occupies 4 bits in TCON register, but there is a gap of 4 4562306a36Sopenharmony_ci * bits (one channel) after channel 0, so channels have different numbering 4662306a36Sopenharmony_ci * when accessing TCON register. 4762306a36Sopenharmony_ci * 4862306a36Sopenharmony_ci * In addition, the location of autoreload bit for channel 4 (TCON channel 5) 4962306a36Sopenharmony_ci * in its set of bits is 2 as opposed to 3 for other channels. 5062306a36Sopenharmony_ci */ 5162306a36Sopenharmony_ci#define TCON_START(chan) (1 << (4 * (chan) + 0)) 5262306a36Sopenharmony_ci#define TCON_MANUALUPDATE(chan) (1 << (4 * (chan) + 1)) 5362306a36Sopenharmony_ci#define TCON_INVERT(chan) (1 << (4 * (chan) + 2)) 5462306a36Sopenharmony_ci#define _TCON_AUTORELOAD(chan) (1 << (4 * (chan) + 3)) 5562306a36Sopenharmony_ci#define _TCON_AUTORELOAD4(chan) (1 << (4 * (chan) + 2)) 5662306a36Sopenharmony_ci#define TCON_AUTORELOAD(chan) \ 5762306a36Sopenharmony_ci ((chan < 5) ? _TCON_AUTORELOAD(chan) : _TCON_AUTORELOAD4(chan)) 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ciDEFINE_SPINLOCK(samsung_pwm_lock); 6062306a36Sopenharmony_ciEXPORT_SYMBOL(samsung_pwm_lock); 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistruct samsung_pwm_clocksource { 6362306a36Sopenharmony_ci void __iomem *base; 6462306a36Sopenharmony_ci const void __iomem *source_reg; 6562306a36Sopenharmony_ci unsigned int irq[SAMSUNG_PWM_NUM]; 6662306a36Sopenharmony_ci struct samsung_pwm_variant variant; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci struct clk *timerclk; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci unsigned int event_id; 7162306a36Sopenharmony_ci unsigned int source_id; 7262306a36Sopenharmony_ci unsigned int tcnt_max; 7362306a36Sopenharmony_ci unsigned int tscaler_div; 7462306a36Sopenharmony_ci unsigned int tdiv; 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci unsigned long clock_count_per_tick; 7762306a36Sopenharmony_ci}; 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_cistatic struct samsung_pwm_clocksource pwm; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_cistatic void samsung_timer_set_prescale(unsigned int channel, u16 prescale) 8262306a36Sopenharmony_ci{ 8362306a36Sopenharmony_ci unsigned long flags; 8462306a36Sopenharmony_ci u8 shift = 0; 8562306a36Sopenharmony_ci u32 reg; 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_ci if (channel >= 2) 8862306a36Sopenharmony_ci shift = TCFG0_PRESCALER1_SHIFT; 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ci spin_lock_irqsave(&samsung_pwm_lock, flags); 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci reg = readl(pwm.base + REG_TCFG0); 9362306a36Sopenharmony_ci reg &= ~(TCFG0_PRESCALER_MASK << shift); 9462306a36Sopenharmony_ci reg |= (prescale - 1) << shift; 9562306a36Sopenharmony_ci writel(reg, pwm.base + REG_TCFG0); 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci spin_unlock_irqrestore(&samsung_pwm_lock, flags); 9862306a36Sopenharmony_ci} 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_cistatic void samsung_timer_set_divisor(unsigned int channel, u8 divisor) 10162306a36Sopenharmony_ci{ 10262306a36Sopenharmony_ci u8 shift = TCFG1_SHIFT(channel); 10362306a36Sopenharmony_ci unsigned long flags; 10462306a36Sopenharmony_ci u32 reg; 10562306a36Sopenharmony_ci u8 bits; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci bits = (fls(divisor) - 1) - pwm.variant.div_base; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci spin_lock_irqsave(&samsung_pwm_lock, flags); 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci reg = readl(pwm.base + REG_TCFG1); 11262306a36Sopenharmony_ci reg &= ~(TCFG1_MUX_MASK << shift); 11362306a36Sopenharmony_ci reg |= bits << shift; 11462306a36Sopenharmony_ci writel(reg, pwm.base + REG_TCFG1); 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci spin_unlock_irqrestore(&samsung_pwm_lock, flags); 11762306a36Sopenharmony_ci} 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_cistatic void samsung_time_stop(unsigned int channel) 12062306a36Sopenharmony_ci{ 12162306a36Sopenharmony_ci unsigned long tcon; 12262306a36Sopenharmony_ci unsigned long flags; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci if (channel > 0) 12562306a36Sopenharmony_ci ++channel; 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_ci spin_lock_irqsave(&samsung_pwm_lock, flags); 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci tcon = readl_relaxed(pwm.base + REG_TCON); 13062306a36Sopenharmony_ci tcon &= ~TCON_START(channel); 13162306a36Sopenharmony_ci writel_relaxed(tcon, pwm.base + REG_TCON); 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_ci spin_unlock_irqrestore(&samsung_pwm_lock, flags); 13462306a36Sopenharmony_ci} 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_cistatic void samsung_time_setup(unsigned int channel, unsigned long tcnt) 13762306a36Sopenharmony_ci{ 13862306a36Sopenharmony_ci unsigned long tcon; 13962306a36Sopenharmony_ci unsigned long flags; 14062306a36Sopenharmony_ci unsigned int tcon_chan = channel; 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci if (tcon_chan > 0) 14362306a36Sopenharmony_ci ++tcon_chan; 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ci spin_lock_irqsave(&samsung_pwm_lock, flags); 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_ci tcon = readl_relaxed(pwm.base + REG_TCON); 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci tcon &= ~(TCON_START(tcon_chan) | TCON_AUTORELOAD(tcon_chan)); 15062306a36Sopenharmony_ci tcon |= TCON_MANUALUPDATE(tcon_chan); 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci writel_relaxed(tcnt, pwm.base + REG_TCNTB(channel)); 15362306a36Sopenharmony_ci writel_relaxed(tcnt, pwm.base + REG_TCMPB(channel)); 15462306a36Sopenharmony_ci writel_relaxed(tcon, pwm.base + REG_TCON); 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci spin_unlock_irqrestore(&samsung_pwm_lock, flags); 15762306a36Sopenharmony_ci} 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_cistatic void samsung_time_start(unsigned int channel, bool periodic) 16062306a36Sopenharmony_ci{ 16162306a36Sopenharmony_ci unsigned long tcon; 16262306a36Sopenharmony_ci unsigned long flags; 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci if (channel > 0) 16562306a36Sopenharmony_ci ++channel; 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci spin_lock_irqsave(&samsung_pwm_lock, flags); 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_ci tcon = readl_relaxed(pwm.base + REG_TCON); 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_ci tcon &= ~TCON_MANUALUPDATE(channel); 17262306a36Sopenharmony_ci tcon |= TCON_START(channel); 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_ci if (periodic) 17562306a36Sopenharmony_ci tcon |= TCON_AUTORELOAD(channel); 17662306a36Sopenharmony_ci else 17762306a36Sopenharmony_ci tcon &= ~TCON_AUTORELOAD(channel); 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci writel_relaxed(tcon, pwm.base + REG_TCON); 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_ci spin_unlock_irqrestore(&samsung_pwm_lock, flags); 18262306a36Sopenharmony_ci} 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_cistatic int samsung_set_next_event(unsigned long cycles, 18562306a36Sopenharmony_ci struct clock_event_device *evt) 18662306a36Sopenharmony_ci{ 18762306a36Sopenharmony_ci /* 18862306a36Sopenharmony_ci * This check is needed to account for internal rounding 18962306a36Sopenharmony_ci * errors inside clockevents core, which might result in 19062306a36Sopenharmony_ci * passing cycles = 0, which in turn would not generate any 19162306a36Sopenharmony_ci * timer interrupt and hang the system. 19262306a36Sopenharmony_ci * 19362306a36Sopenharmony_ci * Another solution would be to set up the clockevent device 19462306a36Sopenharmony_ci * with min_delta = 2, but this would unnecessarily increase 19562306a36Sopenharmony_ci * the minimum sleep period. 19662306a36Sopenharmony_ci */ 19762306a36Sopenharmony_ci if (!cycles) 19862306a36Sopenharmony_ci cycles = 1; 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_ci samsung_time_setup(pwm.event_id, cycles); 20162306a36Sopenharmony_ci samsung_time_start(pwm.event_id, false); 20262306a36Sopenharmony_ci 20362306a36Sopenharmony_ci return 0; 20462306a36Sopenharmony_ci} 20562306a36Sopenharmony_ci 20662306a36Sopenharmony_cistatic int samsung_shutdown(struct clock_event_device *evt) 20762306a36Sopenharmony_ci{ 20862306a36Sopenharmony_ci samsung_time_stop(pwm.event_id); 20962306a36Sopenharmony_ci return 0; 21062306a36Sopenharmony_ci} 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_cistatic int samsung_set_periodic(struct clock_event_device *evt) 21362306a36Sopenharmony_ci{ 21462306a36Sopenharmony_ci samsung_time_stop(pwm.event_id); 21562306a36Sopenharmony_ci samsung_time_setup(pwm.event_id, pwm.clock_count_per_tick - 1); 21662306a36Sopenharmony_ci samsung_time_start(pwm.event_id, true); 21762306a36Sopenharmony_ci return 0; 21862306a36Sopenharmony_ci} 21962306a36Sopenharmony_ci 22062306a36Sopenharmony_cistatic void samsung_clockevent_resume(struct clock_event_device *cev) 22162306a36Sopenharmony_ci{ 22262306a36Sopenharmony_ci samsung_timer_set_prescale(pwm.event_id, pwm.tscaler_div); 22362306a36Sopenharmony_ci samsung_timer_set_divisor(pwm.event_id, pwm.tdiv); 22462306a36Sopenharmony_ci 22562306a36Sopenharmony_ci if (pwm.variant.has_tint_cstat) { 22662306a36Sopenharmony_ci u32 mask = (1 << pwm.event_id); 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci writel(mask | (mask << 5), pwm.base + REG_TINT_CSTAT); 22962306a36Sopenharmony_ci } 23062306a36Sopenharmony_ci} 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_cistatic struct clock_event_device time_event_device = { 23362306a36Sopenharmony_ci .name = "samsung_event_timer", 23462306a36Sopenharmony_ci .features = CLOCK_EVT_FEAT_PERIODIC | 23562306a36Sopenharmony_ci CLOCK_EVT_FEAT_ONESHOT, 23662306a36Sopenharmony_ci .rating = 200, 23762306a36Sopenharmony_ci .set_next_event = samsung_set_next_event, 23862306a36Sopenharmony_ci .set_state_shutdown = samsung_shutdown, 23962306a36Sopenharmony_ci .set_state_periodic = samsung_set_periodic, 24062306a36Sopenharmony_ci .set_state_oneshot = samsung_shutdown, 24162306a36Sopenharmony_ci .tick_resume = samsung_shutdown, 24262306a36Sopenharmony_ci .resume = samsung_clockevent_resume, 24362306a36Sopenharmony_ci}; 24462306a36Sopenharmony_ci 24562306a36Sopenharmony_cistatic irqreturn_t samsung_clock_event_isr(int irq, void *dev_id) 24662306a36Sopenharmony_ci{ 24762306a36Sopenharmony_ci struct clock_event_device *evt = dev_id; 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_ci if (pwm.variant.has_tint_cstat) { 25062306a36Sopenharmony_ci u32 mask = (1 << pwm.event_id); 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci writel(mask | (mask << 5), pwm.base + REG_TINT_CSTAT); 25362306a36Sopenharmony_ci } 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci evt->event_handler(evt); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci return IRQ_HANDLED; 25862306a36Sopenharmony_ci} 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_cistatic void __init samsung_clockevent_init(void) 26162306a36Sopenharmony_ci{ 26262306a36Sopenharmony_ci unsigned long pclk; 26362306a36Sopenharmony_ci unsigned long clock_rate; 26462306a36Sopenharmony_ci unsigned int irq_number; 26562306a36Sopenharmony_ci 26662306a36Sopenharmony_ci pclk = clk_get_rate(pwm.timerclk); 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci samsung_timer_set_prescale(pwm.event_id, pwm.tscaler_div); 26962306a36Sopenharmony_ci samsung_timer_set_divisor(pwm.event_id, pwm.tdiv); 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_ci clock_rate = pclk / (pwm.tscaler_div * pwm.tdiv); 27262306a36Sopenharmony_ci pwm.clock_count_per_tick = clock_rate / HZ; 27362306a36Sopenharmony_ci 27462306a36Sopenharmony_ci time_event_device.cpumask = cpumask_of(0); 27562306a36Sopenharmony_ci clockevents_config_and_register(&time_event_device, 27662306a36Sopenharmony_ci clock_rate, 1, pwm.tcnt_max); 27762306a36Sopenharmony_ci 27862306a36Sopenharmony_ci irq_number = pwm.irq[pwm.event_id]; 27962306a36Sopenharmony_ci if (request_irq(irq_number, samsung_clock_event_isr, 28062306a36Sopenharmony_ci IRQF_TIMER | IRQF_IRQPOLL, "samsung_time_irq", 28162306a36Sopenharmony_ci &time_event_device)) 28262306a36Sopenharmony_ci pr_err("%s: request_irq() failed\n", "samsung_time_irq"); 28362306a36Sopenharmony_ci 28462306a36Sopenharmony_ci if (pwm.variant.has_tint_cstat) { 28562306a36Sopenharmony_ci u32 mask = (1 << pwm.event_id); 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci writel(mask | (mask << 5), pwm.base + REG_TINT_CSTAT); 28862306a36Sopenharmony_ci } 28962306a36Sopenharmony_ci} 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_cistatic void samsung_clocksource_suspend(struct clocksource *cs) 29262306a36Sopenharmony_ci{ 29362306a36Sopenharmony_ci samsung_time_stop(pwm.source_id); 29462306a36Sopenharmony_ci} 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_cistatic void samsung_clocksource_resume(struct clocksource *cs) 29762306a36Sopenharmony_ci{ 29862306a36Sopenharmony_ci samsung_timer_set_prescale(pwm.source_id, pwm.tscaler_div); 29962306a36Sopenharmony_ci samsung_timer_set_divisor(pwm.source_id, pwm.tdiv); 30062306a36Sopenharmony_ci 30162306a36Sopenharmony_ci samsung_time_setup(pwm.source_id, pwm.tcnt_max); 30262306a36Sopenharmony_ci samsung_time_start(pwm.source_id, true); 30362306a36Sopenharmony_ci} 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_cistatic u64 notrace samsung_clocksource_read(struct clocksource *c) 30662306a36Sopenharmony_ci{ 30762306a36Sopenharmony_ci return ~readl_relaxed(pwm.source_reg); 30862306a36Sopenharmony_ci} 30962306a36Sopenharmony_ci 31062306a36Sopenharmony_cistatic struct clocksource samsung_clocksource = { 31162306a36Sopenharmony_ci .name = "samsung_clocksource_timer", 31262306a36Sopenharmony_ci .rating = 250, 31362306a36Sopenharmony_ci .read = samsung_clocksource_read, 31462306a36Sopenharmony_ci .suspend = samsung_clocksource_suspend, 31562306a36Sopenharmony_ci .resume = samsung_clocksource_resume, 31662306a36Sopenharmony_ci .flags = CLOCK_SOURCE_IS_CONTINUOUS, 31762306a36Sopenharmony_ci}; 31862306a36Sopenharmony_ci 31962306a36Sopenharmony_ci/* 32062306a36Sopenharmony_ci * Override the global weak sched_clock symbol with this 32162306a36Sopenharmony_ci * local implementation which uses the clocksource to get some 32262306a36Sopenharmony_ci * better resolution when scheduling the kernel. We accept that 32362306a36Sopenharmony_ci * this wraps around for now, since it is just a relative time 32462306a36Sopenharmony_ci * stamp. (Inspired by U300 implementation.) 32562306a36Sopenharmony_ci */ 32662306a36Sopenharmony_cistatic u64 notrace samsung_read_sched_clock(void) 32762306a36Sopenharmony_ci{ 32862306a36Sopenharmony_ci return samsung_clocksource_read(NULL); 32962306a36Sopenharmony_ci} 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_cistatic int __init samsung_clocksource_init(void) 33262306a36Sopenharmony_ci{ 33362306a36Sopenharmony_ci unsigned long pclk; 33462306a36Sopenharmony_ci unsigned long clock_rate; 33562306a36Sopenharmony_ci 33662306a36Sopenharmony_ci pclk = clk_get_rate(pwm.timerclk); 33762306a36Sopenharmony_ci 33862306a36Sopenharmony_ci samsung_timer_set_prescale(pwm.source_id, pwm.tscaler_div); 33962306a36Sopenharmony_ci samsung_timer_set_divisor(pwm.source_id, pwm.tdiv); 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci clock_rate = pclk / (pwm.tscaler_div * pwm.tdiv); 34262306a36Sopenharmony_ci 34362306a36Sopenharmony_ci samsung_time_setup(pwm.source_id, pwm.tcnt_max); 34462306a36Sopenharmony_ci samsung_time_start(pwm.source_id, true); 34562306a36Sopenharmony_ci 34662306a36Sopenharmony_ci if (pwm.source_id == 4) 34762306a36Sopenharmony_ci pwm.source_reg = pwm.base + 0x40; 34862306a36Sopenharmony_ci else 34962306a36Sopenharmony_ci pwm.source_reg = pwm.base + pwm.source_id * 0x0c + 0x14; 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci sched_clock_register(samsung_read_sched_clock, 35262306a36Sopenharmony_ci pwm.variant.bits, clock_rate); 35362306a36Sopenharmony_ci 35462306a36Sopenharmony_ci samsung_clocksource.mask = CLOCKSOURCE_MASK(pwm.variant.bits); 35562306a36Sopenharmony_ci return clocksource_register_hz(&samsung_clocksource, clock_rate); 35662306a36Sopenharmony_ci} 35762306a36Sopenharmony_ci 35862306a36Sopenharmony_cistatic void __init samsung_timer_resources(void) 35962306a36Sopenharmony_ci{ 36062306a36Sopenharmony_ci clk_prepare_enable(pwm.timerclk); 36162306a36Sopenharmony_ci 36262306a36Sopenharmony_ci pwm.tcnt_max = (1UL << pwm.variant.bits) - 1; 36362306a36Sopenharmony_ci if (pwm.variant.bits == 16) { 36462306a36Sopenharmony_ci pwm.tscaler_div = 25; 36562306a36Sopenharmony_ci pwm.tdiv = 2; 36662306a36Sopenharmony_ci } else { 36762306a36Sopenharmony_ci pwm.tscaler_div = 2; 36862306a36Sopenharmony_ci pwm.tdiv = 1; 36962306a36Sopenharmony_ci } 37062306a36Sopenharmony_ci} 37162306a36Sopenharmony_ci 37262306a36Sopenharmony_ci/* 37362306a36Sopenharmony_ci * PWM master driver 37462306a36Sopenharmony_ci */ 37562306a36Sopenharmony_cistatic int __init _samsung_pwm_clocksource_init(void) 37662306a36Sopenharmony_ci{ 37762306a36Sopenharmony_ci u8 mask; 37862306a36Sopenharmony_ci int channel; 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_ci mask = ~pwm.variant.output_mask & ((1 << SAMSUNG_PWM_NUM) - 1); 38162306a36Sopenharmony_ci channel = fls(mask) - 1; 38262306a36Sopenharmony_ci if (channel < 0) { 38362306a36Sopenharmony_ci pr_crit("failed to find PWM channel for clocksource\n"); 38462306a36Sopenharmony_ci return -EINVAL; 38562306a36Sopenharmony_ci } 38662306a36Sopenharmony_ci pwm.source_id = channel; 38762306a36Sopenharmony_ci 38862306a36Sopenharmony_ci mask &= ~(1 << channel); 38962306a36Sopenharmony_ci channel = fls(mask) - 1; 39062306a36Sopenharmony_ci if (channel < 0) { 39162306a36Sopenharmony_ci pr_crit("failed to find PWM channel for clock event\n"); 39262306a36Sopenharmony_ci return -EINVAL; 39362306a36Sopenharmony_ci } 39462306a36Sopenharmony_ci pwm.event_id = channel; 39562306a36Sopenharmony_ci 39662306a36Sopenharmony_ci samsung_timer_resources(); 39762306a36Sopenharmony_ci samsung_clockevent_init(); 39862306a36Sopenharmony_ci 39962306a36Sopenharmony_ci return samsung_clocksource_init(); 40062306a36Sopenharmony_ci} 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_civoid __init samsung_pwm_clocksource_init(void __iomem *base, 40362306a36Sopenharmony_ci unsigned int *irqs, 40462306a36Sopenharmony_ci const struct samsung_pwm_variant *variant) 40562306a36Sopenharmony_ci{ 40662306a36Sopenharmony_ci pwm.base = base; 40762306a36Sopenharmony_ci memcpy(&pwm.variant, variant, sizeof(pwm.variant)); 40862306a36Sopenharmony_ci memcpy(pwm.irq, irqs, SAMSUNG_PWM_NUM * sizeof(*irqs)); 40962306a36Sopenharmony_ci 41062306a36Sopenharmony_ci pwm.timerclk = clk_get(NULL, "timers"); 41162306a36Sopenharmony_ci if (IS_ERR(pwm.timerclk)) 41262306a36Sopenharmony_ci panic("failed to get timers clock for timer"); 41362306a36Sopenharmony_ci 41462306a36Sopenharmony_ci _samsung_pwm_clocksource_init(); 41562306a36Sopenharmony_ci} 41662306a36Sopenharmony_ci 41762306a36Sopenharmony_ci#ifdef CONFIG_TIMER_OF 41862306a36Sopenharmony_cistatic int __init samsung_pwm_alloc(struct device_node *np, 41962306a36Sopenharmony_ci const struct samsung_pwm_variant *variant) 42062306a36Sopenharmony_ci{ 42162306a36Sopenharmony_ci struct property *prop; 42262306a36Sopenharmony_ci const __be32 *cur; 42362306a36Sopenharmony_ci u32 val; 42462306a36Sopenharmony_ci int i, ret; 42562306a36Sopenharmony_ci 42662306a36Sopenharmony_ci memcpy(&pwm.variant, variant, sizeof(pwm.variant)); 42762306a36Sopenharmony_ci for (i = 0; i < SAMSUNG_PWM_NUM; ++i) 42862306a36Sopenharmony_ci pwm.irq[i] = irq_of_parse_and_map(np, i); 42962306a36Sopenharmony_ci 43062306a36Sopenharmony_ci of_property_for_each_u32(np, "samsung,pwm-outputs", prop, cur, val) { 43162306a36Sopenharmony_ci if (val >= SAMSUNG_PWM_NUM) { 43262306a36Sopenharmony_ci pr_warn("%s: invalid channel index in samsung,pwm-outputs property\n", __func__); 43362306a36Sopenharmony_ci continue; 43462306a36Sopenharmony_ci } 43562306a36Sopenharmony_ci pwm.variant.output_mask |= 1 << val; 43662306a36Sopenharmony_ci } 43762306a36Sopenharmony_ci 43862306a36Sopenharmony_ci pwm.base = of_iomap(np, 0); 43962306a36Sopenharmony_ci if (!pwm.base) { 44062306a36Sopenharmony_ci pr_err("%s: failed to map PWM registers\n", __func__); 44162306a36Sopenharmony_ci return -ENXIO; 44262306a36Sopenharmony_ci } 44362306a36Sopenharmony_ci 44462306a36Sopenharmony_ci pwm.timerclk = of_clk_get_by_name(np, "timers"); 44562306a36Sopenharmony_ci if (IS_ERR(pwm.timerclk)) { 44662306a36Sopenharmony_ci pr_crit("failed to get timers clock for timer\n"); 44762306a36Sopenharmony_ci ret = PTR_ERR(pwm.timerclk); 44862306a36Sopenharmony_ci goto err_clk; 44962306a36Sopenharmony_ci } 45062306a36Sopenharmony_ci 45162306a36Sopenharmony_ci ret = _samsung_pwm_clocksource_init(); 45262306a36Sopenharmony_ci if (ret) 45362306a36Sopenharmony_ci goto err_clocksource; 45462306a36Sopenharmony_ci 45562306a36Sopenharmony_ci return 0; 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_cierr_clocksource: 45862306a36Sopenharmony_ci clk_put(pwm.timerclk); 45962306a36Sopenharmony_ci pwm.timerclk = NULL; 46062306a36Sopenharmony_cierr_clk: 46162306a36Sopenharmony_ci iounmap(pwm.base); 46262306a36Sopenharmony_ci pwm.base = NULL; 46362306a36Sopenharmony_ci 46462306a36Sopenharmony_ci return ret; 46562306a36Sopenharmony_ci} 46662306a36Sopenharmony_ci 46762306a36Sopenharmony_cistatic const struct samsung_pwm_variant s3c24xx_variant = { 46862306a36Sopenharmony_ci .bits = 16, 46962306a36Sopenharmony_ci .div_base = 1, 47062306a36Sopenharmony_ci .has_tint_cstat = false, 47162306a36Sopenharmony_ci .tclk_mask = (1 << 4), 47262306a36Sopenharmony_ci}; 47362306a36Sopenharmony_ci 47462306a36Sopenharmony_cistatic int __init s3c2410_pwm_clocksource_init(struct device_node *np) 47562306a36Sopenharmony_ci{ 47662306a36Sopenharmony_ci return samsung_pwm_alloc(np, &s3c24xx_variant); 47762306a36Sopenharmony_ci} 47862306a36Sopenharmony_ciTIMER_OF_DECLARE(s3c2410_pwm, "samsung,s3c2410-pwm", s3c2410_pwm_clocksource_init); 47962306a36Sopenharmony_ci 48062306a36Sopenharmony_cistatic const struct samsung_pwm_variant s3c64xx_variant = { 48162306a36Sopenharmony_ci .bits = 32, 48262306a36Sopenharmony_ci .div_base = 0, 48362306a36Sopenharmony_ci .has_tint_cstat = true, 48462306a36Sopenharmony_ci .tclk_mask = (1 << 7) | (1 << 6) | (1 << 5), 48562306a36Sopenharmony_ci}; 48662306a36Sopenharmony_ci 48762306a36Sopenharmony_cistatic int __init s3c64xx_pwm_clocksource_init(struct device_node *np) 48862306a36Sopenharmony_ci{ 48962306a36Sopenharmony_ci return samsung_pwm_alloc(np, &s3c64xx_variant); 49062306a36Sopenharmony_ci} 49162306a36Sopenharmony_ciTIMER_OF_DECLARE(s3c6400_pwm, "samsung,s3c6400-pwm", s3c64xx_pwm_clocksource_init); 49262306a36Sopenharmony_ci 49362306a36Sopenharmony_cistatic const struct samsung_pwm_variant s5p64x0_variant = { 49462306a36Sopenharmony_ci .bits = 32, 49562306a36Sopenharmony_ci .div_base = 0, 49662306a36Sopenharmony_ci .has_tint_cstat = true, 49762306a36Sopenharmony_ci .tclk_mask = 0, 49862306a36Sopenharmony_ci}; 49962306a36Sopenharmony_ci 50062306a36Sopenharmony_cistatic int __init s5p64x0_pwm_clocksource_init(struct device_node *np) 50162306a36Sopenharmony_ci{ 50262306a36Sopenharmony_ci return samsung_pwm_alloc(np, &s5p64x0_variant); 50362306a36Sopenharmony_ci} 50462306a36Sopenharmony_ciTIMER_OF_DECLARE(s5p6440_pwm, "samsung,s5p6440-pwm", s5p64x0_pwm_clocksource_init); 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_cistatic const struct samsung_pwm_variant s5p_variant = { 50762306a36Sopenharmony_ci .bits = 32, 50862306a36Sopenharmony_ci .div_base = 0, 50962306a36Sopenharmony_ci .has_tint_cstat = true, 51062306a36Sopenharmony_ci .tclk_mask = (1 << 5), 51162306a36Sopenharmony_ci}; 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_cistatic int __init s5p_pwm_clocksource_init(struct device_node *np) 51462306a36Sopenharmony_ci{ 51562306a36Sopenharmony_ci return samsung_pwm_alloc(np, &s5p_variant); 51662306a36Sopenharmony_ci} 51762306a36Sopenharmony_ciTIMER_OF_DECLARE(s5pc100_pwm, "samsung,s5pc100-pwm", s5p_pwm_clocksource_init); 51862306a36Sopenharmony_ci#endif 519