Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
тип книга.doc
Скачиваний:
6
Добавлен:
01.03.2025
Размер:
1.55 Mб
Скачать

Виртуальная память и кэширование

На первый взгляд может показаться, что виртуальная память и кэширование ни­как не связаны, но на самом деле эти механизмы похожи. При наличии вир­туальной памяти вся программа хранится на диске и разбивается на страницы фиксированного размера. Некоторое подмножество этих страниц находится в основной памяти. Если программа главным образом использует страницы из ос­новной памяти, то ошибки отсутствия страницы будут встречаться редко, и про­грамма сможет работать быстро. При кэшировании вся программа хранится в ос­новной памяти и разбивается на блоки фиксированного размера. Некоторое подмножество этих блоков находится в кэш-памяти. Если же программа главным образом использует блоки из кэш-памяти, то кэш-промахи будут происходить редко, и программа сможет работать быстро. Как видим, виртуальная память и кэш-память идентичны, разве что работают они на разных уровнях иерархии.

Естественно, виртуальная память и кэш-память кое в чем различаются. Кэш-промахи обрабатываются аппаратно, а ошибки отсутствия страниц обраба­тываются операционной системой. Блоки кэш-памяти обычно гораздо меньше страниц (сравните: 64 байта против 8 Кбайт). Кроме того, таблицы страниц ин­дексируются по старшим битам виртуального адреса, а кэш-память индексирует­ся по младшим битам адреса памяти. Тем не менее важно понимать, что разли­чие здесь только в реализации.

Виртуальные команды ввода-вывода

Как вам уже известно, наборы команд на уровне архитектуры команд и на уров­не микроархитектуры совершенно разные. Различаются не только сами коман­ды, но и их форматы, а некоторые совпадения совершенно случайны.

Набор команд уровня операционной системы содержит большую часть команд уровня архитектуры команд, а также несколько новых очень важных команд. В то же время некоторые ненужные команды на уровне операционной системы не поддерживаются. Ввод-вывод — это одна из областей, в которых эти два уров­ня отличаются очень значительно. Причина таких различий проста. Во-первых, пользователь, способный выполнять команды ввода-вывода уровня архитектуры команд, может считать конфиденциальную информацию, хранящуюся где-ни­будь в «дебрях» операционной системы, и вообще потенциально представляет для системы угрозу. Во-вторых, нормальный программист совершенно не горит желанием реализовать ввод-вывод на уровне архитектуры команд, поскольку это слишком сложно и утомительно. Вместо этого для ввода-вывода можно просто установить определенные поля и биты в ряде регистров устройств, затем подож­дать, пока операция закончится, и проверить, что произошло. Обычно биты ре­гистров дисковых устройств позволяют обнаруживать следующие ошибки:

+ аппаратура диска не смогла выполнить позиционирование;

+ несуществующий элемент памяти определен как буфер;

+ процесс ввода-вывода с диска (на диск) начался до того, как закончился предыдущий;

+ ошибка синхронизации при считывании;

+ обращение к несуществующему диску;

+ обращение к несуществующему цилиндру;

+ обращение к несуществующему сектору;

+ несоответствие контрольных сумм при считывании;

+ ошибка проверки записи.

При наличии одной из этих ошибок устанавливается соответствующий бит в регистре устройства.

Файлы

Один из способов виртуального ввода-вывода — использование абстракции под названием файл. Файл состоит из последовательности байтов, записанных на устройство ввода-вывода. Если устройство ввода-вывода является устройством хранения информации (например, диском), то файл можно считать обратно. Ес­ли устройство не является устройством хранения информации (например, это принтер), то файл оттуда считать нельзя. На диске может храниться множество файлов, в каждом из которых содержатся данные определенного типа, например, изображение, электронная таблица или текст. Файлы имеют разную длину и об­ладают разными свойствами. Эта абстракция позволяет легко организовать вир­туальный ввод-вывод.

Для операционной системы файл является просто последовательностью бай­тов. Ввод-вывод файла осуществляется путем системных вызовов открытия, чте­ния, записи и закрытия файлов. Перед тем как считывать файл, его нужно от­крыть. Процесс открытия файла позволяет операционной системе найти файл на диске и передать в память информацию, необходимую для доступа к этому файлу.

После открытия файла его можно считывать. Системный вызов для считыва­ния должен иметь, как минимум, следующие параметры:

+ информацию о том, какой именно открытый файл нужно считывать;

♦ указатель на буфер в памяти, в который нужно поместить данные;

+ число считываемых байтов.

Данный системный вызов помещает требующиеся данные в буфер. Обычно он возвращает число считанных байтов. Это число может быть меньше запро­шенного числа (например, нельзя считать 2000 байт из файла размером 1000 байт).

С каждым открытым файлом связан указатель, который сообщает, какой байт должен считываться следующим. После команды read указатель дополняется числом считанных байтов, поэтому последовательные команды read считывают последовательные блоки данных из файла. Обычно этот указатель можно уста­новить на особое значение, чтобы программы могли получить доступ к любой части файла. Когда программа заканчивает считывание файла, она может закрыть его и сообщить операционной системе, что больше не будет использовать этот файл. Тогда операционная система сможет освободить пространство в структу­ре, в которой хранилась информация об этом файле.

В операционных системах для универсальных вычислительных машин файл представляет собой более сложную структуру. Здесь файл может быть последо­вательностью логических записей, каждая из которых имеет строго определенную структуру. Например, логическая запись может представлять собой структуру данных, состоящую из пяти полей: двух символьных строк, «Имя» и «Началь­ник», двух целых чисел, «Отдел» и «Комната», и одного логического числа «Пол женский». Некоторые операционные системы различают файлы, в которых все элементы имеют одинаковую структуру, и файлы, содержащие разные типы данных.

Основная виртуальная команда ввода считывает следующую запись из нуж­ного файла и помещает ее в основную память, начиная с определенного адреса, как показано на рис. 6.17. Чтобы выполнить эту операцию, виртуальная команда должна получить сведения о том, какой файл считывать и куда в памяти поместить запись. Часто существуют параметры для чтения некоторой записи, кото­рые определяются либо по ее месту в файле, либо по ее ключу.


Основная виртуальная команда вывода записывает логическую запись из па­мяти в файл. Последовательные команды write выполняют последовательные логические записи в файл.