blob: 8a590c6191faca7cebe69302a8e3bf409ea0002d [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Graeme Russ8c63d472009-02-24 21:14:45 +11002/*
3 * (C) Copyright 2002
Albert ARIBAUDfa82f872011-08-04 18:45:45 +02004 * Daniel Engström, Omicron Ceti AB, <daniel@omicron.se>
Graeme Russ8c63d472009-02-24 21:14:45 +11005 */
6
Graeme Russ8c63d472009-02-24 21:14:45 +11007#include <asm/io.h>
8#include <asm/i8254.h>
Simon Glassdbfb6c02023-07-30 21:01:48 -06009#include <asm/ibmpc.h>
Tom Rinie51478b2024-04-27 08:10:56 -060010#include <linux/errno.h>
Graeme Russ8c63d472009-02-24 21:14:45 +110011
Simon Glass79a5be82019-02-16 20:24:58 -070012#define TIMER1_VALUE 18 /* 15.6us */
13#define BEEP_FREQUENCY_HZ 440
14#define SYSCTL_PORTB 0x61
15#define PORTB_BEEP_ENABLE 0x3
16
17static void i8254_set_beep_freq(uint frequency_hz)
18{
19 uint countdown;
20
21 countdown = PIT_TICK_RATE / frequency_hz;
22
23 outb(countdown & 0xff, PIT_BASE + PIT_T2);
24 outb((countdown >> 8) & 0xff, PIT_BASE + PIT_T2);
25}
Graeme Russ8c63d472009-02-24 21:14:45 +110026
Bin Mengda3fe242015-10-22 19:13:30 -070027int i8254_init(void)
Graeme Russ8c63d472009-02-24 21:14:45 +110028{
Simon Glassd0b6f242013-04-17 16:13:39 +000029 /*
Bin Mengbffeed02015-10-22 19:13:29 -070030 * Initialize counter 1, used to refresh request signal.
31 * This is required for legacy purpose as some codes like
32 * vgabios utilizes counter 1 to provide delay functionality.
33 */
34 outb(PIT_CMD_CTR1 | PIT_CMD_LOW | PIT_CMD_MODE2,
35 PIT_BASE + PIT_COMMAND);
36 outb(TIMER1_VALUE, PIT_BASE + PIT_T1);
37
38 /*
Bin Meng0a2ea022015-10-22 19:13:28 -070039 * Initialize counter 2, used to drive the speaker.
40 * To start a beep, set both bit0 and bit1 of port 0x61.
41 * To stop it, clear both bit0 and bit1 of port 0x61.
Graeme Russ8c63d472009-02-24 21:14:45 +110042 */
Graeme Russ83088af2011-11-08 02:33:15 +000043 outb(PIT_CMD_CTR2 | PIT_CMD_BOTH | PIT_CMD_MODE3,
Bin Meng0a2ea022015-10-22 19:13:28 -070044 PIT_BASE + PIT_COMMAND);
Simon Glass79a5be82019-02-16 20:24:58 -070045 i8254_set_beep_freq(BEEP_FREQUENCY_HZ);
Graeme Russ8c63d472009-02-24 21:14:45 +110046
Graeme Russ8c63d472009-02-24 21:14:45 +110047 return 0;
48}
Simon Glass79a5be82019-02-16 20:24:58 -070049
50int i8254_enable_beep(uint frequency_hz)
51{
52 if (!frequency_hz)
53 return -EINVAL;
54
Bin Meng7d0a53a2019-02-26 01:52:19 -080055 /* make sure i8254 is setup correctly before generating beeps */
56 outb(PIT_CMD_CTR2 | PIT_CMD_BOTH | PIT_CMD_MODE3,
57 PIT_BASE + PIT_COMMAND);
58
Simon Glass79a5be82019-02-16 20:24:58 -070059 i8254_set_beep_freq(frequency_hz);
60 setio_8(SYSCTL_PORTB, PORTB_BEEP_ENABLE);
61
62 return 0;
63}
64
65void i8254_disable_beep(void)
66{
67 clrio_8(SYSCTL_PORTB, PORTB_BEEP_ENABLE);
68}