Форум программистов «Весельчак У»
  *
Добро пожаловать, Гость. Пожалуйста, войдите или зарегистрируйтесь.
Вам не пришло письмо с кодом активации?

  • Рекомендуем проверить настройки временной зоны в вашем профиле (страница "Внешний вид форума", пункт "Часовой пояс:").
  • У нас больше нет рассылок. Если вам приходят письма от наших бывших рассылок mail.ru и subscribe.ru, то знайте, что это не мы рассылаем.
   Начало  
Наши сайты
Помощь Поиск Календарь Почта Войти Регистрация  
 
Страниц: [1]   Вниз
  Печать  
Автор Тема: PCI+DMA+UserMode  (Прочитано 5971 раз)
0 Пользователей и 1 Гость смотрят эту тему.
Eugene
Гость
« : 27-11-2004 21:40 » 

Есть следующая проблема. Только учусь писать драйвера. Использую Driver Studio. Есть плата PCI. Требуется получать с неё данные (большие объёмы) по DMA и  писать их в файл. Вопрос в следующем: Как в драйвере получить физический адрес буфера, куда собственно и собираюсь складывать данные, если буфер создан в пользовательском приложении. И ещё как сигнализировать пользовательскому прложению о том, что данные получены. А можно ли прямо из драйвера писать в файл.

Зараннее спасибо.
Записан
Rulik
Помогающий

ru
Offline Offline

« Ответ #1 : 30-11-2004 06:31 » 

Писать из драйвера в файл можно!

Смотри функции ZwWriteFile, ZwCreateFile.
Записан
Серж
Гость
« Ответ #2 : 30-11-2004 08:01 » 

Eugene, для получения физического адреса можно использовать функцию MmGetPhysicalAddress, но чтобы избежать головной боли, лучше создавать DMA-буфера в драйвере, используя класс KDmaAdapter и его метод
AllocateCommonBuffer. Для общения с приложением можно использовать объекты синхронизации: Event, Mutex, Semaphor, Timer, причем создавать их можно, как в пользовательской моде, так и в драйвере, только техника использования немного разная. Обо всем этом написано в документации к DriverStudio.
Записан
Eugene
Гость
« Ответ #3 : 05-12-2004 01:27 » 

Серж, спасибо за ответ. Но есть вот такой вопрос.
В данный момент ситуация такая: драйвер написан человеком, которого уже нет, а мне нужно разобраться и внести различные исправления.

Сделано на данный момент так:
1\ используем плату PCI (bus master device);
2\ плата передаёт данные блоками по 64 kb;
3\ по завершении передачи бока плата генерит прерывание, сигнализирующее о том, что передача блока завершена.
4\ чтобы плата начала передавать данные в неё пишется сколько байтов мы хотим получить, и начальный физический
адрес памяти, куда это всё ложить. Область памяти выделенная под данные, как я понимаю должна быть непрерывной.
5\ Поступает запрос (IOCTL) из приложения на чтение, например 4Mb. В драйвере выделяется Common Buffer размером 64kb. Его адрес и подсовывается плате. По приходу прерывания(т.е. получили 64kb) данные копируются из него в некий другой буффер большего размера. После этого адрес Common Buffera опять подсовыается плате и передача данных продолжается и опять копирование в другой буфер и так до тех пор пока не будет накоплено 4Mb. После этого в пользовательское приложение передаётся Event и пользовательское приложение передаёт с помощью IOCTL запроса(METHOD_NEITHER) свой буфер, уже куда копируются данныё из драйверного буфера(RtlCopyMemory).

Вопрос: Как лучше и правильнее всё это сделать? Мне кажется, что здесь какие-то копирования лишние и вообще не очень всё правильно сделано. Если не очень сложно поподробнее, пожалуйста.
Записан
Страниц: [1]   Вверх
  Печать  
 

Powered by SMF 1.1.21 | SMF © 2015, Simple Machines