

## به نام خدا

# آزمایشگاه سیستم عامل

# پروژه چهارم: همگام سازی

طراحان: على عباسي، على عطااللهي





#### مقدمه

در این پروژه با سازوکارهای همگامسازی اسیستم عاملها آشنا خواهید شد. با توجه به این که سیستم عامل XV6 از ریسههای سطح کاربر پشتیبانی نمی کند، همگامسازی در سطح پردازهها مطرح خواهد بود. همچنین به علت عدم پشتیبانی از حافظه مشترک در این سیستم عامل، همگامسازی در سطح هسته صورت خواهد گرفت. به همین سبب مختصری راجع به این قسم از همگامسازی توضیح داده خواهد شد.

 $^{\scriptscriptstyle 1}$  Synchronization Mechanisms

\_

## ضرورت همگام سازی در هسته سیستم عامل ها

هسته سیستم عاملها دارای مسیرهای کنترلی  $^2$  مختلفی میباشد. به طور کلی، دنباله دستورالعملهای اجرا شده توسط هسته جهت مدیریت فراخوانی سیستمی، وقفه یا استثنا این مسیرها را تشکیل میدهند. در این میان برخی از سیستم عاملها دارای هسته با ورود مجدد  $^3$  می باشند. بدین معنی که مسیرهای کنترلی این هستهها قابلیت اجرای همروند دارند. تمامی سیستم عامل های مدرن کنونی این قابلیت را دارند. مثلا ممکن است برنامه سطح کاربر در میانه اجرای فراخوانی سیستمی در هسته باشد که وقفه هایی رخ دهد. به این ترتیب در حین اجرای یک مسیر کنترلی در هسته (اجرای کد فراخوانی سیستمی)، مسیر کنترلی دیگری در هسته (اجرای کد مدیریت وقفه) شروع به اجرا نموده و به نوعی دوباره ورود به هسته صورت می پذیرد. وجود همزمان چند مسیر کنترلی در هسته می تواند منجر به وجود شرایط مسابقه برای دسترسی به حالت مشترک هسته گردد. به این ترتیب، اجرای صحیح کد هسته مستلزم همگام سازی باید ماهیتهای مختلف کدهای اجرایی هسته لحاظ گردد.

هر مسیر کنترلی هسته در یک متن خاص اجرا می گردد. اگر کد هسته به طور مستقیم یا غیرمستقیم توسط برنامه سطح کاربر اجرا گردد، در متن پردازه و اجرا می گردد. در حالی که کدی که در نتیجه وقفه اجرا می گردد در متن وقفه است. به این ترتیب فراخوانی سیستمی و استثناها در متن پردازه فراخواننده هستند. در حالی که وقفه در متن وقفه اجرا می گردد. به طور کلی در سیستم عاملها کدهای وقفه قابل مسدود شدن نیستند. ماهیت این کدهای اجرایی به این صورت است که باید در اسرع وقت اجرا شده و لذا قابل زمانبندی توسط زمانبند نیز نیستند. به این ترتیب سازوکار همگامسازی آنها نباید منجر به مسدود شدن آنها گردد، مثلا از قفلهای چرخشی آستفاده گردد یا در پردازنده های تک هسته ای وقفه غیر فعال گردد.

<sup>&</sup>lt;sup>2</sup> Control Path

<sup>&</sup>lt;sup>3</sup> Reentrant Kernel

<sup>&</sup>lt;sup>4</sup> Concurrent

<sup>&</sup>lt;sup>5</sup> Process Context

<sup>&</sup>lt;sup>6</sup> Interrupt Context

<sup>&</sup>lt;sup>7</sup> Spin Locks

#### همگامسازی در xv6

قفل گذاری در هسته xv6 توسط دو سری تابع صورت می گیرد. دسته اول شامل توابع xv6 خط xv6 و فقل گذاری در هسته xv6 می توسط دو سری تابع صورت می گیرد. دسته اول شامل توابع xv6 مناطر v6 انتظار v6 می تولد. این قفل ها منجر به انتظار مشغول v6 شده و در حین اجرای ناحیه بحرانی وقفه را نیز غیرفعال می کنند.

- 1) علت غیرفعال کردن وقفه در هنگام استفاده از این نوع قفل چیست؟ چرا ممکن است CPU با مشکل deadlock رو به رو شود؟
  - 2) توابع pushcli و popcli به چه منظور استفاده شده و چه تفاوتی با sti و sti دارند؟
  - 3) چرا قفل مذکور در سیستم های تک هسته ای مناسب نیست؟ روی کد توضیح دهید.
- 4) در مجموعه دستورات RISC-V، دستوری با نام amoswap وجود دارد. دلیل تعریف و نحوه کار آن را توضیح دهید.

دسته دوم شامل توابع acquiresleep (خط 4621) و releasesleep (خط 4633) بوده که مشکل انتظار مشغول را حل نموده و امکان تعامل میان پردازه ها را نیز فراهم می کنند. تفاوت اصلی توابع این دسته نسبت به دسته قبل این است که در صورت عدم امکان در اختیار گرفتن قفل، از تلاش دست کشیده و پردازنده را رها می کنند.

- 5) مختصری راجع به تعامل میان پردازه ها توسط دو تابع مذکور توضیح دهید.
- 6) حالات مختلف پردازهها در xv6 را توضیح دهید. تابع sched چه وظیفه ای دارد؟

یک مشکل در توابع دسته دوم عدم وجود نگهدارنده و قفل است. به این ترتیب حتی پردازهای که قفل را در اختیار ندارد می تواند با فراخوانی تابع releasesleep قفل را آزاد نماید.

7) تغییری در توابع دسته دوم داده تا تنها پردازه صاحب قفل، قادر به آزادسازی آن باشد. قفل معادل در هسته لینوکس را به طور مختصر معرفی نمایید.

<sup>&</sup>lt;sup>8</sup> Busy Waiting

<sup>9</sup> Owner

8) روشی دیگر برای نوشتن برنامهها استفاده از الگوریتم های lock-free است. مختصری راجع به آن ها توضیح داده و از مزایا و معایب آنها نسبت به برنامه نویسی با lock بگویید.

#### ییادهسازی متغیرهای مختص هر هسته پردازنده

یکی از روشهای افزایش کارایی در پردازندهها استفاده از حافظه نهان  $^{10}$  است. حافظههای نهان در سطوح  $^{11}$  مختلف وجود داشته و می توانند محلی  $^{12}$  یا مشترک  $^{13}$  باشند. به عنوان مثال، معمولا حافظه نهان سطح یک  $^{14}$  مختص هر هسته پردازنده بوده و لذا محلی است. بدین ترتیب هرگاه پردازهای در یک متغیر حافظه بنویسد، مقادیر نگهداری شده برای این متغیر در حافظههای نهان سطح دیگر هستههای پردازنده را نامعتبر  $^{15}$  می نماید.

الف) روشی جهت حل این مشکل در سطح سخت افزار وجود دارد. مختصرا آن را توضیح دهید.

معتبرسازی مقادیر حافظه نهان محلی، سربار قابل توجهی داشته و میتواند کارایی سیستم را پایین بیاورد.

ب) همان طور که در اسلایدهای معرفی پروژه ذکر شده است، یکی از روشهای همگام سازی استفاده از قفلهایی مرسوم به قفل بلیت است. این قفلها را از منظر مشکل مذکور در بالا بررسی نمایید.

در بسیاری از کاربردها می توان با استفاده از متغیرهای مختص هر هسته، مشکل را حل نمود. به این ترتیب که به جز در موارد ضروری، دسترسی و بهروزرسانی را در نسخه مختص هسته جاری از متغیر انجام می دهند. بدین ترتیب با کاهش تعداد معتبرسازی، سربار کاهش می یابد.

ج) چگونه می توان در لینوکس داده های مختص هر هسته را در زمان کامپایل تعریف نمود؟

با استفاده از این روش، یک فراخوانی سیستمی تعریف نمایید که تعداد فراخوانیهای سیستمی اجرا شده در یک بار کاری را روی یک سیستم چهار هستهای برمی گرداند.

<sup>&</sup>lt;sup>10</sup> Cache

<sup>11</sup> Levels

<sup>12</sup> Local

<sup>13</sup> Shared

<sup>14</sup> L1 Cache

<sup>15</sup> Invalid

باید به تعداد کافی پردازه ایجاد نمایید که در فایلهایی مینویسند. همچنین جهت اطمینان از صحت عملکرد باید یک نسخه مشترک میان همه هستهها تعریف شده و با مقدار برگشتی مقایسه گردد. دقت کنید همواره باید متغیر مربوط به هسته در حال اجرا، بهروزرسانی گردد (راهنمایی: روش پیادهسازی قفل چرخشی XV6 میتواند راهگشا باشد).

نیازی به ترازبندی حافظه نهان نیست.

## پیاده سازی سازوکار همگام سازی با قابلیت اولویت دادن

در این قسمت میخواهیم سازوکاری برای همگامسازی پیادهسازی کنیم که در آن پردازهها میتوانند برای ورود به ناحیه بحرانی دارای اولویت باشند. حال پردازهها میتوانند برای ورود به ناحیه بحرانی درخواست دهند و پردازه با اولویت بالاتر ابتدا وارد ناحیه بحرانی میشود. بعد از آن نیز پرداژهها که در یک صف دارای اولویت هستند به ترتیب وارد میشوند. در زمانی که یک پردازه قفل را در اختیار دارد و در ناحیه بحرانی هست نیز ممکن است پردازههای جدیدی با اولویتهای متفاوت وارد شوند که در این صورت ترتیب صف اولویت به صورت پویا عوض می شود.

برای پیاده سازی فرض کنید که اولویت پردازه های متفاوت شماره پردازه آن می باشد. شما نیاز است که برنامه سطح کاربری بنویسید که درستی پیاده سازی را نشان دهد. برای این کار چند پردازه در سطح کاربر ایجاد کنید که هر پردازه به دنبال ورود به ناحیه بحرانی و دریافت قفل می باشد.

حال در ناحیه بحرانی یک کار زمان بر انجام دهید تا پردازه های دیگر خود را به صف اضافه کنند. در هر مرحله نمایش دهید که پردازه با چه اولویتی وارد ناحیه بحرانی شد. همچنین صف اولویت را نیز نمایش دهید.

\_

<sup>&</sup>lt;sup>16</sup> Cache Alignment

آیا این پیادهسازی ممکن است که دچار گرسنگی شود؟ راه حلی برای برطرف کردن این مشکل ارائه دهید. روش ارائه شده توسط شما باید بتواند شرایطی را که قفلها دارای اولویت یکسان می باشند را نیز پوشش دهد ( نیازی به پیاده سازی برای این قسمت نیست).

یک نوع پیاده سازی همگام سازی توسط قفل بلیت <sup>17</sup> انجام می شود. آن را بررسی کنید و تفاوت های آن با روش همگام سازی بالا را بیان کنید.

#### ساير نكات:

- مدیریت حافظه مناسب در پروژه از نکات مهم پیادهسازی است.
- از لاگهای مناسب در پیاده سازی استفاده نمایید تا تست و اشکالزدایی کد ساده تر شود. واضح است که استفاده بیش از حد از آنها باعث سردرگمی خواهد شد.
  - فقط فایلهای تغییر یافته و یا افزوده شده را به صورت ZIP بارگذاری نمایید.
  - همه افراد باید به پروژه مسلط باشند و نمره تمامی اعضای گروه لزوما یکسان نخواهد بود.
    - در صورت تشخیص تقلب، نمره هر دو گروه صفر در نظر گرفته خواهد شد.
      - فصل 4 و انتهای فصل 5 کتاب xv6 می تواند مفید باشد.
    - هرگونه سوال در مورد پروژه را از طریق ایمیلهای طراحان می توانید مطرح نمایید.

aliataollahi40@gmail.com

موفق باشيد

<sup>&</sup>lt;sup>17</sup> Ticket Lock