pcspkr.c 1.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. /* vim: tabstop=4 shiftwidth=4 noexpandtab
  2. * This file is part of ToaruOS and is released under the terms
  3. * of the NCSA / University of Illinois License - see LICENSE.md
  4. * Copyright (C) 2014-2018 K. Lange
  5. */
  6. #include <kernel/module.h>
  7. #include <kernel/printf.h>
  8. #include <kernel/mod/shell.h>
  9. static void note(int length, int freq) {
  10. uint32_t div = 11931800 / freq;
  11. uint8_t t;
  12. outportb(0x43, 0xb6);
  13. outportb(0x42, (uint8_t)(div));
  14. outportb(0x42, (uint8_t)(div >> 8));
  15. t = inportb(0x61);
  16. outportb(0x61, t | 0x3);
  17. unsigned long s, ss;
  18. relative_time(0, length * 10, &s, &ss);
  19. sleep_until((process_t *)current_process, s, ss);
  20. switch_task(0);
  21. t = inportb(0x61) & 0xFC;
  22. outportb(0x61, t);
  23. }
  24. struct spkr {
  25. int length;
  26. int frequency;
  27. };
  28. static uint32_t write_spkr(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer) {
  29. if (!size % (sizeof(struct spkr))) {
  30. return 0;
  31. }
  32. struct spkr * s = (struct spkr *)buffer;
  33. while ((uintptr_t)s < (uintptr_t)buffer + size) {
  34. note(s->length, s->frequency);
  35. s++;
  36. }
  37. return (uintptr_t)s - (uintptr_t)buffer;
  38. }
  39. static fs_node_t * spkr_device_create(void) {
  40. fs_node_t * fnode = malloc(sizeof(fs_node_t));
  41. memset(fnode, 0x00, sizeof(fs_node_t));
  42. sprintf(fnode->name, "spkr");
  43. fnode->mask = 0666; /* TODO need a speaker group */
  44. fnode->flags = FS_CHARDEVICE;
  45. fnode->write = write_spkr;
  46. return fnode;
  47. }
  48. static int init(void) {
  49. fs_node_t * node = spkr_device_create();
  50. vfs_mount("/dev/spkr", node);
  51. return 0;
  52. }
  53. static int fini(void) {
  54. return 0;
  55. }
  56. MODULE_DEF(pcspkr, init, fini);