18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * m68k beeper driver for Linux 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (c) 2002 Richard Zidlicky 68c2ecf20Sopenharmony_ci * Copyright (c) 2002 Vojtech Pavlik 78c2ecf20Sopenharmony_ci * Copyright (c) 1992 Orest Zborowski 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci 118c2ecf20Sopenharmony_ci#include <linux/kernel.h> 128c2ecf20Sopenharmony_ci#include <linux/module.h> 138c2ecf20Sopenharmony_ci#include <linux/init.h> 148c2ecf20Sopenharmony_ci#include <linux/input.h> 158c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 168c2ecf20Sopenharmony_ci#include <asm/machdep.h> 178c2ecf20Sopenharmony_ci#include <asm/io.h> 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_ciMODULE_AUTHOR("Richard Zidlicky <rz@linux-m68k.org>"); 208c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("m68k beeper driver"); 218c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_cistatic struct platform_device *m68kspkr_platform_device; 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_cistatic int m68kspkr_event(struct input_dev *dev, unsigned int type, unsigned int code, int value) 268c2ecf20Sopenharmony_ci{ 278c2ecf20Sopenharmony_ci unsigned int count = 0; 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_ci if (type != EV_SND) 308c2ecf20Sopenharmony_ci return -1; 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_ci switch (code) { 338c2ecf20Sopenharmony_ci case SND_BELL: if (value) value = 1000; 348c2ecf20Sopenharmony_ci case SND_TONE: break; 358c2ecf20Sopenharmony_ci default: return -1; 368c2ecf20Sopenharmony_ci } 378c2ecf20Sopenharmony_ci 388c2ecf20Sopenharmony_ci if (value > 20 && value < 32767) 398c2ecf20Sopenharmony_ci count = 1193182 / value; 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_ci mach_beep(count, -1); 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_ci return 0; 448c2ecf20Sopenharmony_ci} 458c2ecf20Sopenharmony_ci 468c2ecf20Sopenharmony_cistatic int m68kspkr_probe(struct platform_device *dev) 478c2ecf20Sopenharmony_ci{ 488c2ecf20Sopenharmony_ci struct input_dev *input_dev; 498c2ecf20Sopenharmony_ci int err; 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci input_dev = input_allocate_device(); 528c2ecf20Sopenharmony_ci if (!input_dev) 538c2ecf20Sopenharmony_ci return -ENOMEM; 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci input_dev->name = "m68k beeper"; 568c2ecf20Sopenharmony_ci input_dev->phys = "m68k/generic"; 578c2ecf20Sopenharmony_ci input_dev->id.bustype = BUS_HOST; 588c2ecf20Sopenharmony_ci input_dev->id.vendor = 0x001f; 598c2ecf20Sopenharmony_ci input_dev->id.product = 0x0001; 608c2ecf20Sopenharmony_ci input_dev->id.version = 0x0100; 618c2ecf20Sopenharmony_ci input_dev->dev.parent = &dev->dev; 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_ci input_dev->evbit[0] = BIT_MASK(EV_SND); 648c2ecf20Sopenharmony_ci input_dev->sndbit[0] = BIT_MASK(SND_BELL) | BIT_MASK(SND_TONE); 658c2ecf20Sopenharmony_ci input_dev->event = m68kspkr_event; 668c2ecf20Sopenharmony_ci 678c2ecf20Sopenharmony_ci err = input_register_device(input_dev); 688c2ecf20Sopenharmony_ci if (err) { 698c2ecf20Sopenharmony_ci input_free_device(input_dev); 708c2ecf20Sopenharmony_ci return err; 718c2ecf20Sopenharmony_ci } 728c2ecf20Sopenharmony_ci 738c2ecf20Sopenharmony_ci platform_set_drvdata(dev, input_dev); 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci return 0; 768c2ecf20Sopenharmony_ci} 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_cistatic int m68kspkr_remove(struct platform_device *dev) 798c2ecf20Sopenharmony_ci{ 808c2ecf20Sopenharmony_ci struct input_dev *input_dev = platform_get_drvdata(dev); 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_ci input_unregister_device(input_dev); 838c2ecf20Sopenharmony_ci /* turn off the speaker */ 848c2ecf20Sopenharmony_ci m68kspkr_event(NULL, EV_SND, SND_BELL, 0); 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci return 0; 878c2ecf20Sopenharmony_ci} 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_cistatic void m68kspkr_shutdown(struct platform_device *dev) 908c2ecf20Sopenharmony_ci{ 918c2ecf20Sopenharmony_ci /* turn off the speaker */ 928c2ecf20Sopenharmony_ci m68kspkr_event(NULL, EV_SND, SND_BELL, 0); 938c2ecf20Sopenharmony_ci} 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_cistatic struct platform_driver m68kspkr_platform_driver = { 968c2ecf20Sopenharmony_ci .driver = { 978c2ecf20Sopenharmony_ci .name = "m68kspkr", 988c2ecf20Sopenharmony_ci }, 998c2ecf20Sopenharmony_ci .probe = m68kspkr_probe, 1008c2ecf20Sopenharmony_ci .remove = m68kspkr_remove, 1018c2ecf20Sopenharmony_ci .shutdown = m68kspkr_shutdown, 1028c2ecf20Sopenharmony_ci}; 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_cistatic int __init m68kspkr_init(void) 1058c2ecf20Sopenharmony_ci{ 1068c2ecf20Sopenharmony_ci int err; 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci if (!mach_beep) { 1098c2ecf20Sopenharmony_ci printk(KERN_INFO "m68kspkr: no lowlevel beep support\n"); 1108c2ecf20Sopenharmony_ci return -ENODEV; 1118c2ecf20Sopenharmony_ci } 1128c2ecf20Sopenharmony_ci 1138c2ecf20Sopenharmony_ci err = platform_driver_register(&m68kspkr_platform_driver); 1148c2ecf20Sopenharmony_ci if (err) 1158c2ecf20Sopenharmony_ci return err; 1168c2ecf20Sopenharmony_ci 1178c2ecf20Sopenharmony_ci m68kspkr_platform_device = platform_device_alloc("m68kspkr", -1); 1188c2ecf20Sopenharmony_ci if (!m68kspkr_platform_device) { 1198c2ecf20Sopenharmony_ci err = -ENOMEM; 1208c2ecf20Sopenharmony_ci goto err_unregister_driver; 1218c2ecf20Sopenharmony_ci } 1228c2ecf20Sopenharmony_ci 1238c2ecf20Sopenharmony_ci err = platform_device_add(m68kspkr_platform_device); 1248c2ecf20Sopenharmony_ci if (err) 1258c2ecf20Sopenharmony_ci goto err_free_device; 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_ci return 0; 1288c2ecf20Sopenharmony_ci 1298c2ecf20Sopenharmony_ci err_free_device: 1308c2ecf20Sopenharmony_ci platform_device_put(m68kspkr_platform_device); 1318c2ecf20Sopenharmony_ci err_unregister_driver: 1328c2ecf20Sopenharmony_ci platform_driver_unregister(&m68kspkr_platform_driver); 1338c2ecf20Sopenharmony_ci 1348c2ecf20Sopenharmony_ci return err; 1358c2ecf20Sopenharmony_ci} 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_cistatic void __exit m68kspkr_exit(void) 1388c2ecf20Sopenharmony_ci{ 1398c2ecf20Sopenharmony_ci platform_device_unregister(m68kspkr_platform_device); 1408c2ecf20Sopenharmony_ci platform_driver_unregister(&m68kspkr_platform_driver); 1418c2ecf20Sopenharmony_ci} 1428c2ecf20Sopenharmony_ci 1438c2ecf20Sopenharmony_cimodule_init(m68kspkr_init); 1448c2ecf20Sopenharmony_cimodule_exit(m68kspkr_exit); 145